From 5cf484206181dd6119058e8959ed570973c26b62 Mon Sep 17 00:00:00 2001 From: Nayeon Keum Date: Fri, 29 Sep 2023 22:54:05 +0900 Subject: [PATCH] [feat/#15] Add: init kubespray --- kubespray/Dockerfile | 45 + kubespray/Makefile | 7 + kubespray/README.md | 283 ++ kubespray/SECURITY_CONTACTS | 15 + kubespray/Vagrantfile | 290 ++ kubespray/ansible.cfg | 22 + kubespray/cluster.yml | 3 + .../aws_iam/kubernetes-master-policy.json | 27 + .../aws_iam/kubernetes-master-role.json | 10 + .../aws_iam/kubernetes-minion-policy.json | 45 + .../aws_iam/kubernetes-minion-role.json | 10 + .../aws_inventory/kubespray-aws-inventory.py | 81 + .../contrib/aws_inventory/requirements.txt | 1 + kubespray/contrib/azurerm/.gitignore | 2 + kubespray/contrib/azurerm/README.md | 67 + kubespray/contrib/azurerm/apply-rg.sh | 19 + kubespray/contrib/azurerm/clear-rg.sh | 14 + .../contrib/azurerm/generate-inventory.sh | 18 + .../contrib/azurerm/generate-inventory.yml | 6 + .../contrib/azurerm/generate-inventory_2.yml | 6 + .../contrib/azurerm/generate-templates.yml | 6 + kubespray/contrib/azurerm/group_vars/all | 51 + .../roles/generate-inventory/tasks/main.yml | 15 + .../generate-inventory/templates/inventory.j2 | 33 + .../roles/generate-inventory_2/tasks/main.yml | 31 + .../templates/inventory.j2 | 33 + .../templates/loadbalancer_vars.j2 | 8 + .../generate-templates/defaults/main.yml | 37 + .../roles/generate-templates/tasks/main.yml | 25 + .../templates/availability-sets.json | 30 + .../generate-templates/templates/bastion.json | 106 + .../templates/clear-rg.json | 8 + .../generate-templates/templates/masters.json | 198 + .../generate-templates/templates/minions.json | 115 + .../generate-templates/templates/network.json | 109 + .../generate-templates/templates/storage.json | 19 + kubespray/contrib/dind/README.md | 177 + kubespray/contrib/dind/dind-cluster.yaml | 11 + .../contrib/dind/group_vars/all/all.yaml | 3 + .../contrib/dind/group_vars/all/distro.yaml | 41 + kubespray/contrib/dind/hosts | 15 + kubespray/contrib/dind/kubespray-dind.yaml | 22 + kubespray/contrib/dind/requirements.txt | 1 + .../dind/roles/dind-cluster/tasks/main.yaml | 73 + .../dind/roles/dind-host/tasks/main.yaml | 87 + .../templates/inventory_builder.sh.j2 | 3 + kubespray/contrib/dind/run-test-distros.sh | 93 + .../dind/test-most_distros-some_CNIs.env | 11 + .../test-some_distros-kube_router_combo.env | 6 + .../dind/test-some_distros-most_CNIs.env | 8 + .../contrib/inventory_builder/inventory.py | 480 ++ .../inventory_builder/requirements.txt | 3 + kubespray/contrib/inventory_builder/setup.cfg | 3 + kubespray/contrib/inventory_builder/setup.py | 29 + .../inventory_builder/test-requirements.txt | 3 + .../inventory_builder/tests/test_inventory.py | 595 +++ kubespray/contrib/inventory_builder/tox.ini | 34 + kubespray/contrib/kvm-setup/README.md | 11 + kubespray/contrib/kvm-setup/group_vars/all | 2 + kubespray/contrib/kvm-setup/kvm-setup.yml | 9 + .../kvm-setup/roles/kvm-setup/tasks/main.yml | 30 + .../roles/kvm-setup/tasks/sysctl.yml | 46 + .../kvm-setup/roles/kvm-setup/tasks/user.yml | 47 + kubespray/contrib/misc/clusteradmin-rbac.yml | 15 + kubespray/contrib/mitogen/mitogen.yml | 51 + .../network-storage/glusterfs/README.md | 92 + .../network-storage/glusterfs/glusterfs.yml | 29 + .../network-storage/glusterfs/group_vars | 1 + .../glusterfs/inventory.example | 43 + .../glusterfs/roles/bootstrap-os | 1 + .../glusterfs/roles/glusterfs/README.md | 50 + .../roles/glusterfs/client/defaults/main.yml | 11 + .../roles/glusterfs/client/meta/main.yml | 30 + .../roles/glusterfs/client/tasks/main.yml | 21 + .../glusterfs/client/tasks/setup-Debian.yml | 24 + .../glusterfs/client/tasks/setup-RedHat.yml | 14 + .../roles/glusterfs/server/defaults/main.yml | 13 + .../roles/glusterfs/server/meta/main.yml | 30 + .../roles/glusterfs/server/tasks/main.yml | 113 + .../glusterfs/server/tasks/setup-Debian.yml | 26 + .../glusterfs/server/tasks/setup-RedHat.yml | 15 + .../glusterfs/server/templates/test-file.txt | 1 + .../roles/glusterfs/server/vars/Debian.yml | 2 + .../roles/glusterfs/server/vars/RedHat.yml | 2 + .../kubernetes-pv/ansible/tasks/main.yaml | 23 + .../glusterfs-kubernetes-endpoint-svc.json.j2 | 12 + .../glusterfs-kubernetes-endpoint.json.j2 | 23 + .../templates/glusterfs-kubernetes-pv.yml.j2 | 14 + .../roles/kubernetes-pv/meta/main.yaml | 3 + .../contrib/network-storage/heketi/README.md | 27 + .../heketi/heketi-tear-down.yml | 11 + .../contrib/network-storage/heketi/heketi.yml | 12 + .../heketi/inventory.yml.sample | 33 + .../network-storage/heketi/requirements.txt | 1 + .../heketi/roles/prepare/tasks/main.yml | 24 + .../heketi/roles/provision/defaults/main.yml | 1 + .../heketi/roles/provision/handlers/main.yml | 3 + .../roles/provision/tasks/bootstrap.yml | 64 + .../provision/tasks/bootstrap/deploy.yml | 27 + .../provision/tasks/bootstrap/storage.yml | 33 + .../provision/tasks/bootstrap/tear-down.yml | 14 + .../provision/tasks/bootstrap/topology.yml | 27 + .../provision/tasks/bootstrap/volumes.yml | 41 + .../heketi/roles/provision/tasks/cleanup.yml | 4 + .../roles/provision/tasks/glusterfs.yml | 44 + .../roles/provision/tasks/glusterfs/label.yml | 19 + .../heketi/roles/provision/tasks/heketi.yml | 34 + .../heketi/roles/provision/tasks/main.yml | 30 + .../heketi/roles/provision/tasks/secret.yml | 45 + .../heketi/roles/provision/tasks/storage.yml | 15 + .../roles/provision/tasks/storageclass.yml | 26 + .../heketi/roles/provision/tasks/topology.yml | 26 + .../templates/glusterfs-daemonset.json.j2 | 149 + .../templates/heketi-bootstrap.json.j2 | 138 + .../templates/heketi-deployment.json.j2 | 164 + .../templates/heketi-service-account.json.j2 | 7 + .../templates/heketi-storage.json.j2 | 54 + .../roles/provision/templates/heketi.json.j2 | 44 + .../provision/templates/storageclass.yml.j2 | 12 + .../provision/templates/topology.json.j2 | 34 + .../roles/tear-down-disks/defaults/main.yml | 2 + .../roles/tear-down-disks/tasks/main.yml | 52 + .../heketi/roles/tear-down/tasks/main.yml | 51 + kubespray/contrib/offline/README.md | 65 + kubespray/contrib/offline/docker-daemon.json | 1 + kubespray/contrib/offline/generate_list.sh | 33 + kubespray/contrib/offline/generate_list.yml | 22 + .../manage-offline-container-images.sh | 172 + .../contrib/offline/manage-offline-files.sh | 44 + kubespray/contrib/offline/nginx.conf | 39 + kubespray/contrib/offline/registries.conf | 8 + kubespray/contrib/os-services/os-services.yml | 5 + .../roles/prepare/defaults/main.yml | 2 + .../os-services/roles/prepare/tasks/main.yml | 23 + .../contrib/packaging/rpm/kubespray.spec | 62 + kubespray/contrib/terraform/OWNERS | 5 + kubespray/contrib/terraform/aws/.gitignore | 3 + kubespray/contrib/terraform/aws/README.md | 162 + .../terraform/aws/create-infrastructure.tf | 179 + .../terraform/aws/credentials.tfvars.example | 8 + .../terraform/aws/docs/aws_kubespray.png | Bin 0 -> 116455 bytes .../contrib/terraform/aws/modules/iam/main.tf | 141 + .../terraform/aws/modules/iam/outputs.tf | 7 + .../terraform/aws/modules/iam/variables.tf | 3 + .../contrib/terraform/aws/modules/nlb/main.tf | 41 + .../terraform/aws/modules/nlb/outputs.tf | 11 + .../terraform/aws/modules/nlb/variables.tf | 30 + .../contrib/terraform/aws/modules/vpc/main.tf | 137 + .../terraform/aws/modules/vpc/outputs.tf | 19 + .../terraform/aws/modules/vpc/variables.tf | 27 + kubespray/contrib/terraform/aws/output.tf | 27 + .../aws/sample-inventory/cluster.tfvars | 59 + .../terraform/aws/sample-inventory/group_vars | 1 + .../terraform/aws/templates/inventory.tpl | 27 + .../contrib/terraform/aws/terraform.tfvars | 43 + .../terraform/aws/terraform.tfvars.example | 33 + kubespray/contrib/terraform/aws/variables.tf | 125 + kubespray/contrib/terraform/equinix/README.md | 246 + kubespray/contrib/terraform/equinix/hosts | 1 + .../contrib/terraform/equinix/kubespray.tf | 57 + kubespray/contrib/terraform/equinix/output.tf | 15 + .../contrib/terraform/equinix/provider.tf | 17 + .../equinix/sample-inventory/cluster.tfvars | 35 + .../equinix/sample-inventory/group_vars | 1 + .../contrib/terraform/equinix/variables.tf | 56 + .../contrib/terraform/exoscale/README.md | 152 + .../contrib/terraform/exoscale/default.tfvars | 65 + kubespray/contrib/terraform/exoscale/main.tf | 49 + .../modules/kubernetes-cluster/main.tf | 193 + .../modules/kubernetes-cluster/output.tf | 31 + .../templates/cloud-init.tmpl | 52 + .../modules/kubernetes-cluster/variables.tf | 42 + .../modules/kubernetes-cluster/versions.tf | 9 + .../contrib/terraform/exoscale/output.tf | 15 + .../exoscale/sample-inventory/cluster.tfvars | 65 + .../exoscale/sample-inventory/group_vars | 1 + .../exoscale/templates/inventory.tpl | 19 + .../contrib/terraform/exoscale/variables.tf | 46 + .../contrib/terraform/exoscale/versions.tf | 15 + kubespray/contrib/terraform/gcp/README.md | 104 + .../terraform/gcp/generate-inventory.sh | 76 + kubespray/contrib/terraform/gcp/main.tf | 39 + .../gcp/modules/kubernetes-cluster/main.tf | 421 ++ .../gcp/modules/kubernetes-cluster/output.tf | 27 + .../modules/kubernetes-cluster/variables.tf | 86 + kubespray/contrib/terraform/gcp/output.tf | 15 + kubespray/contrib/terraform/gcp/tfvars.json | 63 + kubespray/contrib/terraform/gcp/variables.tf | 108 + kubespray/contrib/terraform/group_vars | 1 + kubespray/contrib/terraform/hetzner/README.md | 121 + .../contrib/terraform/hetzner/default.tfvars | 46 + kubespray/contrib/terraform/hetzner/main.tf | 57 + .../kubernetes-cluster-flatcar/main.tf | 144 + .../kubernetes-cluster-flatcar/outputs.tf | 29 + .../templates/machine.yaml.tmpl | 19 + .../kubernetes-cluster-flatcar/variables.tf | 60 + .../kubernetes-cluster-flatcar/versions.tf | 14 + .../modules/kubernetes-cluster/main.tf | 122 + .../modules/kubernetes-cluster/output.tf | 27 + .../templates/cloud-init.tmpl | 16 + .../modules/kubernetes-cluster/variables.tf | 44 + .../modules/kubernetes-cluster/versions.tf | 9 + kubespray/contrib/terraform/hetzner/output.tf | 7 + .../hetzner/sample-inventory/cluster.tfvars | 46 + .../hetzner/sample-inventory/group_vars | 1 + .../terraform/hetzner/templates/inventory.tpl | 19 + .../contrib/terraform/hetzner/variables.tf | 56 + .../contrib/terraform/hetzner/versions.tf | 12 + .../contrib/terraform/nifcloud/.gitignore | 5 + .../contrib/terraform/nifcloud/README.md | 137 + .../terraform/nifcloud/generate-inventory.sh | 64 + kubespray/contrib/terraform/nifcloud/main.tf | 36 + .../modules/kubernetes-cluster/main.tf | 301 ++ .../modules/kubernetes-cluster/outputs.tf | 48 + .../templates/userdata.tftpl | 45 + .../modules/kubernetes-cluster/terraform.tf | 9 + .../modules/kubernetes-cluster/variables.tf | 81 + .../contrib/terraform/nifcloud/output.tf | 3 + .../nifcloud/sample-inventory/cluster.tfvars | 22 + .../nifcloud/sample-inventory/group_vars | 1 + .../contrib/terraform/nifcloud/terraform.tf | 9 + .../contrib/terraform/nifcloud/variables.tf | 77 + .../contrib/terraform/openstack/.gitignore | 5 + .../contrib/terraform/openstack/README.md | 786 +++ kubespray/contrib/terraform/openstack/hosts | 1 + .../contrib/terraform/openstack/kubespray.tf | 130 + .../compute/ansible_bastion_template.txt | 1 + .../openstack/modules/compute/main.tf | 971 ++++ .../compute/templates/cloudinit.yaml.tmpl | 54 + .../openstack/modules/compute/variables.tf | 235 + .../openstack/modules/compute/versions.tf | 8 + .../terraform/openstack/modules/ips/main.tf | 46 + .../openstack/modules/ips/outputs.tf | 25 + .../openstack/modules/ips/variables.tf | 27 + .../openstack/modules/ips/versions.tf | 11 + .../openstack/modules/network/main.tf | 34 + .../openstack/modules/network/outputs.tf | 15 + .../openstack/modules/network/variables.tf | 21 + .../openstack/modules/network/versions.tf | 8 + .../openstack/sample-inventory/cluster.tfvars | 89 + .../openstack/sample-inventory/group_vars | 1 + .../contrib/terraform/openstack/variables.tf | 342 ++ .../contrib/terraform/openstack/versions.tf | 9 + kubespray/contrib/terraform/terraform.py | 474 ++ kubespray/contrib/terraform/upcloud/README.md | 143 + .../terraform/upcloud/cluster-settings.tfvars | 148 + kubespray/contrib/terraform/upcloud/main.tf | 73 + .../modules/kubernetes-cluster/main.tf | 558 +++ .../modules/kubernetes-cluster/output.tf | 24 + .../modules/kubernetes-cluster/variables.tf | 105 + .../modules/kubernetes-cluster/versions.tf | 10 + kubespray/contrib/terraform/upcloud/output.tf | 12 + .../upcloud/sample-inventory/cluster.tfvars | 149 + .../upcloud/sample-inventory/group_vars | 1 + .../terraform/upcloud/templates/inventory.tpl | 17 + .../contrib/terraform/upcloud/variables.tf | 144 + .../contrib/terraform/upcloud/versions.tf | 10 + kubespray/contrib/terraform/vsphere/README.md | 128 + .../contrib/terraform/vsphere/default.tfvars | 38 + kubespray/contrib/terraform/vsphere/main.tf | 100 + .../modules/kubernetes-cluster/main.tf | 149 + .../modules/kubernetes-cluster/output.tf | 15 + .../templates/cloud-init.tpl | 6 + .../kubernetes-cluster/templates/metadata.tpl | 14 + .../templates/vapp-cloud-init.tpl | 24 + .../modules/kubernetes-cluster/variables.tf | 43 + .../modules/kubernetes-cluster/versions.tf | 9 + kubespray/contrib/terraform/vsphere/output.tf | 31 + .../vsphere/sample-inventory/cluster.tfvars | 33 + .../vsphere/sample-inventory/group_vars | 1 + .../terraform/vsphere/templates/inventory.tpl | 17 + .../contrib/terraform/vsphere/variables.tf | 101 + .../contrib/terraform/vsphere/versions.tf | 9 + kubespray/docs/_sidebar.md | 67 + kubespray/docs/amazonlinux.md | 15 + kubespray/docs/ansible.md | 307 ++ kubespray/docs/ansible_collection.md | 38 + kubespray/docs/arch.md | 17 + kubespray/docs/aws-ebs-csi.md | 87 + kubespray/docs/aws.md | 93 + kubespray/docs/azure-csi.md | 128 + kubespray/docs/azure.md | 123 + kubespray/docs/bootstrap-os.md | 61 + kubespray/docs/calico.md | 426 ++ .../docs/calico_peer_example/new-york.yml | 12 + kubespray/docs/calico_peer_example/paris.yml | 12 + kubespray/docs/centos.md | 12 + kubespray/docs/cert_manager.md | 196 + kubespray/docs/cgroups.md | 72 + kubespray/docs/ci-setup.md | 211 + kubespray/docs/ci.md | 57 + kubespray/docs/cilium.md | 244 + kubespray/docs/cinder-csi.md | 102 + kubespray/docs/cloud.md | 13 + kubespray/docs/cni.md | 8 + kubespray/docs/comparisons.md | 26 + kubespray/docs/containerd.md | 142 + kubespray/docs/cri-o.md | 74 + kubespray/docs/debian.md | 41 + kubespray/docs/dns-stack.md | 313 ++ kubespray/docs/docker.md | 99 + kubespray/docs/downloads.md | 41 + .../docs/encrypting-secret-data-at-rest.md | 22 + kubespray/docs/equinix-metal.md | 100 + kubespray/docs/etcd.md | 52 + kubespray/docs/fcos.md | 69 + .../docs/figures/kubespray-calico-rr.png | Bin 0 -> 40710 bytes .../docs/figures/loadbalancer_localhost.png | Bin 0 -> 58266 bytes kubespray/docs/flannel.md | 51 + kubespray/docs/flatcar.md | 14 + kubespray/docs/gcp-lb.md | 20 + kubespray/docs/gcp-pd-csi.md | 77 + kubespray/docs/getting-started.md | 144 + kubespray/docs/gvisor.md | 16 + kubespray/docs/ha-mode.md | 158 + kubespray/docs/hardening.md | 137 + kubespray/docs/img/kubelet-hardening.png | Bin 0 -> 1547778 bytes kubespray/docs/img/kubernetes-logo.png | Bin 0 -> 6954 bytes .../alb_ingress_controller.md | 43 + .../docs/ingress_controller/ingress_nginx.md | 203 + kubespray/docs/integration.md | 188 + kubespray/docs/kata-containers.md | 101 + kubespray/docs/kube-ovn.md | 55 + kubespray/docs/kube-router.md | 79 + kubespray/docs/kube-vip.md | 72 + .../kubernetes-apps/cephfs_provisioner.md | 73 + .../local_volume_provisioner.md | 131 + .../docs/kubernetes-apps/rbd_provisioner.md | 79 + kubespray/docs/kubernetes-apps/registry.md | 244 + kubespray/docs/kubernetes-reliability.md | 108 + kubespray/docs/kylinlinux.md | 11 + kubespray/docs/large-deployments.md | 52 + kubespray/docs/macvlan.md | 41 + kubespray/docs/metallb.md | 222 + kubespray/docs/mirror.md | 66 + kubespray/docs/mitogen.md | 30 + kubespray/docs/multus.md | 74 + kubespray/docs/netcheck.md | 41 + kubespray/docs/nodes.md | 185 + kubespray/docs/ntp.md | 50 + kubespray/docs/offline-environment.md | 152 + kubespray/docs/openeuler.md | 11 + kubespray/docs/openstack.md | 158 + kubespray/docs/opensuse.md | 17 + kubespray/docs/port-requirements.md | 70 + kubespray/docs/proxy.md | 29 + kubespray/docs/recover-control-plane.md | 42 + kubespray/docs/rhel.md | 34 + kubespray/docs/roadmap.md | 3 + .../docs/setting-up-your-first-cluster.md | 642 +++ kubespray/docs/test_cases.md | 33 + kubespray/docs/uoslinux.md | 9 + kubespray/docs/upgrades.md | 418 ++ .../upgrades/migrate_docker2containerd.md | 106 + kubespray/docs/vagrant.md | 164 + kubespray/docs/vars.md | 322 ++ kubespray/docs/vsphere-csi.md | 102 + kubespray/docs/vsphere.md | 134 + kubespray/docs/weave.md | 79 + .../extra_playbooks/files/get_cinder_pvs.sh | 2 + kubespray/extra_playbooks/inventory | 1 + .../migrate_openstack_provider.yml | 30 + kubespray/extra_playbooks/roles | 1 + .../extra_playbooks/upgrade-only-k8s.yml | 61 + .../extra_playbooks/wait-for-cloud-init.yml | 6 + kubespray/inventory/local/group_vars | 1 + kubespray/inventory/local/hosts.ini | 14 + .../moaroom-cluster/group_vars/all/all.yml | 139 + .../moaroom-cluster/group_vars/all/aws.yml | 9 + .../moaroom-cluster/group_vars/all/azure.yml | 40 + .../group_vars/all/containerd.yml | 46 + .../moaroom-cluster/group_vars/all/coreos.yml | 2 + .../moaroom-cluster/group_vars/all/cri-o.yml | 6 + .../moaroom-cluster/group_vars/all/docker.yml | 59 + .../moaroom-cluster/group_vars/all/etcd.yml | 16 + .../moaroom-cluster/group_vars/all/gcp.yml | 10 + .../moaroom-cluster/group_vars/all/hcloud.yml | 22 + .../group_vars/all/huaweicloud.yml | 17 + .../moaroom-cluster/group_vars/all/oci.yml | 28 + .../group_vars/all/offline.yml | 106 + .../group_vars/all/openstack.yml | 50 + .../group_vars/all/upcloud.yml | 24 + .../group_vars/all/vsphere.yml | 32 + .../moaroom-cluster/group_vars/etcd.yml | 26 + .../group_vars/k8s_cluster/addons.yml | 261 + .../group_vars/k8s_cluster/k8s-cluster.yml | 382 ++ .../group_vars/k8s_cluster/k8s-net-calico.yml | 131 + .../group_vars/k8s_cluster/k8s-net-cilium.yml | 264 + .../k8s_cluster/k8s-net-flannel.yml | 18 + .../k8s_cluster/k8s-net-kube-ovn.yml | 63 + .../k8s_cluster/k8s-net-kube-router.yml | 64 + .../k8s_cluster/k8s-net-macvlan.yml | 6 + .../group_vars/k8s_cluster/k8s-net-weave.yml | 64 + .../inventory/moaroom-cluster/inventory.ini | 42 + .../kube-controller-manager+merge.yaml | 8 + .../patches/kube-scheduler+merge.yaml | 8 + .../inventory/sample/group_vars/all/all.yml | 139 + .../inventory/sample/group_vars/all/aws.yml | 9 + .../inventory/sample/group_vars/all/azure.yml | 40 + .../sample/group_vars/all/containerd.yml | 46 + .../sample/group_vars/all/coreos.yml | 2 + .../inventory/sample/group_vars/all/cri-o.yml | 6 + .../sample/group_vars/all/docker.yml | 59 + .../inventory/sample/group_vars/all/etcd.yml | 16 + .../inventory/sample/group_vars/all/gcp.yml | 10 + .../sample/group_vars/all/hcloud.yml | 22 + .../sample/group_vars/all/huaweicloud.yml | 17 + .../inventory/sample/group_vars/all/oci.yml | 28 + .../sample/group_vars/all/offline.yml | 106 + .../sample/group_vars/all/openstack.yml | 50 + .../sample/group_vars/all/upcloud.yml | 24 + .../sample/group_vars/all/vsphere.yml | 32 + .../inventory/sample/group_vars/etcd.yml | 26 + .../sample/group_vars/k8s_cluster/addons.yml | 261 + .../group_vars/k8s_cluster/k8s-cluster.yml | 382 ++ .../group_vars/k8s_cluster/k8s-net-calico.yml | 131 + .../group_vars/k8s_cluster/k8s-net-cilium.yml | 264 + .../k8s_cluster/k8s-net-flannel.yml | 18 + .../k8s_cluster/k8s-net-kube-ovn.yml | 63 + .../k8s_cluster/k8s-net-kube-router.yml | 64 + .../k8s_cluster/k8s-net-macvlan.yml | 6 + .../group_vars/k8s_cluster/k8s-net-weave.yml | 64 + kubespray/inventory/sample/inventory.ini | 38 + .../kube-controller-manager+merge.yaml | 8 + .../sample/patches/kube-scheduler+merge.yaml | 8 + kubespray/library/kube.py | 1 + kubespray/logo/LICENSE | 1 + kubespray/logo/OWNERS | 4 + kubespray/logo/logo-clear.png | Bin 0 -> 4679 bytes kubespray/logo/logo-clear.svg | 80 + kubespray/logo/logo-dark.png | Bin 0 -> 6360 bytes kubespray/logo/logo-dark.svg | 83 + kubespray/logo/logo-text-clear.png | Bin 0 -> 13074 bytes kubespray/logo/logo-text-clear.svg | 107 + kubespray/logo/logo-text-dark.png | Bin 0 -> 13384 bytes kubespray/logo/logo-text-dark.svg | 110 + kubespray/logo/logo-text-mixed.png | Bin 0 -> 16076 bytes kubespray/logo/logo-text-mixed.svg | 110 + kubespray/logo/logos.pdf | Bin 0 -> 288304 bytes kubespray/logo/usage_guidelines.md | 16 + kubespray/meta/runtime.yml | 2 + kubespray/pipeline.Dockerfile | 58 + kubespray/playbooks/ansible_version.yml | 34 + kubespray/playbooks/cluster.yml | 133 + kubespray/playbooks/facts.yml | 41 + kubespray/playbooks/legacy_groups.yml | 47 + kubespray/playbooks/recover_control_plane.yml | 39 + kubespray/playbooks/remove_node.yml | 54 + kubespray/playbooks/reset.yml | 46 + kubespray/playbooks/scale.yml | 115 + kubespray/playbooks/upgrade_cluster.yml | 168 + kubespray/plugins/modules/kube.py | 366 ++ kubespray/recover-control-plane.yml | 3 + kubespray/remove-node.yml | 3 + kubespray/requirements.txt | 9 + kubespray/reset.yml | 3 + kubespray/roles/adduser/defaults/main.yml | 27 + .../adduser/molecule/default/converge.yml | 10 + .../adduser/molecule/default/molecule.yml | 23 + .../molecule/default/tests/test_default.py | 43 + kubespray/roles/adduser/tasks/main.yml | 16 + kubespray/roles/adduser/vars/coreos.yml | 8 + kubespray/roles/adduser/vars/debian.yml | 15 + kubespray/roles/adduser/vars/redhat.yml | 15 + .../bastion-ssh-config/defaults/main.yml | 2 + .../molecule/default/converge.yml | 15 + .../molecule/default/molecule.yml | 31 + .../molecule/default/tests/test_default.py | 40 + .../roles/bastion-ssh-config/tasks/main.yml | 22 + .../templates/ssh-bastion.conf.j2 | 18 + .../roles/bootstrap-os/defaults/main.yml | 32 + .../roles/bootstrap-os/files/bootstrap.sh | 46 + .../roles/bootstrap-os/handlers/main.yml | 4 + .../molecule/default/converge.yml | 6 + .../molecule/default/molecule.yml | 53 + .../molecule/default/tests/test_default.py | 11 + .../bootstrap-os/tasks/bootstrap-amazon.yml | 13 + .../bootstrap-os/tasks/bootstrap-centos.yml | 118 + .../tasks/bootstrap-clearlinux.yml | 16 + .../bootstrap-os/tasks/bootstrap-coreos.yml | 37 + .../bootstrap-os/tasks/bootstrap-debian.yml | 76 + .../tasks/bootstrap-fedora-coreos.yml | 46 + .../bootstrap-os/tasks/bootstrap-fedora.yml | 36 + .../bootstrap-os/tasks/bootstrap-flatcar.yml | 37 + .../bootstrap-os/tasks/bootstrap-opensuse.yml | 85 + .../bootstrap-os/tasks/bootstrap-redhat.yml | 113 + kubespray/roles/bootstrap-os/tasks/main.yml | 109 + .../containerd-common/defaults/main.yml | 17 + .../containerd-common/meta/main.yml | 2 + .../containerd-common/tasks/main.yml | 31 + .../containerd-common/vars/amazon.yml | 2 + .../containerd-common/vars/suse.yml | 2 + .../containerd/defaults/main.yml | 106 + .../containerd/handlers/main.yml | 21 + .../containerd/handlers/reset.yml | 1 + .../container-engine/containerd/meta/main.yml | 6 + .../containerd/molecule/default/converge.yml | 9 + .../containerd/molecule/default/molecule.yml | 47 + .../containerd/molecule/default/prepare.yml | 29 + .../molecule/default/tests/test_default.py | 55 + .../containerd/tasks/main.yml | 140 + .../containerd/tasks/reset.yml | 40 + .../containerd/templates/config.toml.j2 | 88 + .../templates/containerd.service.j2 | 45 + .../containerd/templates/hosts.toml.j2 | 8 + .../containerd/templates/http-proxy.conf.j2 | 2 + .../containerd/vars/debian.yml | 7 + .../containerd/vars/ubuntu.yml | 7 + .../cri-dockerd/handlers/main.yml | 35 + .../cri-dockerd/meta/main.yml | 4 + .../cri-dockerd/molecule/default/converge.yml | 9 + .../molecule/default/files/10-mynet.conf | 17 + .../molecule/default/files/container.json | 10 + .../molecule/default/files/sandbox.json | 10 + .../cri-dockerd/molecule/default/molecule.yml | 39 + .../cri-dockerd/molecule/default/prepare.yml | 48 + .../molecule/default/tests/test_default.py | 19 + .../cri-dockerd/tasks/main.yml | 28 + .../templates/cri-dockerd.service.j2 | 44 + .../templates/cri-dockerd.socket.j2 | 12 + .../container-engine/cri-o/defaults/main.yml | 99 + .../container-engine/cri-o/files/mounts.conf | 1 + .../container-engine/cri-o/handlers/main.yml | 16 + .../container-engine/cri-o/meta/main.yml | 5 + .../cri-o/molecule/default/converge.yml | 9 + .../molecule/default/files/10-mynet.conf | 17 + .../molecule/default/files/container.json | 10 + .../cri-o/molecule/default/files/sandbox.json | 10 + .../cri-o/molecule/default/molecule.yml | 57 + .../cri-o/molecule/default/prepare.yml | 53 + .../molecule/default/tests/test_default.py | 35 + .../container-engine/cri-o/tasks/cleanup.yaml | 120 + .../container-engine/cri-o/tasks/main.yaml | 214 + .../container-engine/cri-o/tasks/reset.yml | 87 + .../cri-o/tasks/setup-amazon.yaml | 38 + .../cri-o/templates/config.json.j2 | 17 + .../cri-o/templates/crio.conf.j2 | 384 ++ .../cri-o/templates/http-proxy.conf.j2 | 2 + .../cri-o/templates/registry.conf.j2 | 13 + .../cri-o/templates/unqualified.conf.j2 | 10 + .../container-engine/crictl/handlers/main.yml | 12 + .../container-engine/crictl/tasks/crictl.yml | 22 + .../container-engine/crictl/tasks/main.yml | 3 + .../crictl/templates/crictl.yaml.j2 | 4 + .../container-engine/crun/tasks/main.yml | 12 + .../docker-storage/defaults/main.yml | 19 + .../files/install_container_storage_setup.sh | 23 + .../docker-storage/tasks/main.yml | 48 + .../templates/docker-storage-setup.j2 | 35 + .../container-engine/docker/defaults/main.yml | 64 + .../docker/files/cleanup-docker-orphans.sh | 38 + .../container-engine/docker/handlers/main.yml | 32 + .../container-engine/docker/meta/main.yml | 5 + .../docker/tasks/docker_plugin.yml | 8 + .../container-engine/docker/tasks/main.yml | 180 + .../docker/tasks/pre-upgrade.yml | 36 + .../container-engine/docker/tasks/reset.yml | 106 + .../docker/tasks/set_facts_dns.yml | 66 + .../container-engine/docker/tasks/systemd.yml | 68 + .../docker/templates/docker-dns.conf.j2 | 6 + .../docker/templates/docker-options.conf.j2 | 11 + .../templates/docker-orphan-cleanup.conf.j2 | 2 + .../docker/templates/docker.service.j2 | 51 + .../docker/templates/fedora_docker.repo.j2 | 7 + .../docker/templates/http-proxy.conf.j2 | 2 + .../docker/templates/rh_docker.repo.j2 | 10 + .../container-engine/docker/vars/amazon.yml | 15 + .../docker/vars/clearlinux.yml | 4 + .../docker/vars/debian-bookworm.yml | 48 + .../container-engine/docker/vars/debian.yml | 61 + .../container-engine/docker/vars/fedora.yml | 49 + .../container-engine/docker/vars/kylin.yml | 53 + .../container-engine/docker/vars/redhat-7.yml | 52 + .../container-engine/docker/vars/redhat.yml | 52 + .../container-engine/docker/vars/suse.yml | 6 + .../container-engine/docker/vars/ubuntu.yml | 61 + .../docker/vars/uniontech.yml | 54 + .../gvisor/molecule/default/converge.yml | 11 + .../molecule/default/files/10-mynet.conf | 17 + .../molecule/default/files/container.json | 10 + .../molecule/default/files/sandbox.json | 10 + .../gvisor/molecule/default/molecule.yml | 39 + .../gvisor/molecule/default/prepare.yml | 49 + .../molecule/default/tests/test_default.py | 29 + .../container-engine/gvisor/tasks/main.yml | 20 + .../container-engine/kata-containers/OWNERS | 6 + .../kata-containers/defaults/main.yml | 10 + .../molecule/default/converge.yml | 11 + .../molecule/default/files/10-mynet.conf | 17 + .../molecule/default/files/container.json | 10 + .../molecule/default/files/sandbox.json | 10 + .../molecule/default/molecule.yml | 39 + .../molecule/default/prepare.yml | 49 + .../molecule/default/tests/test_default.py | 37 + .../kata-containers/tasks/main.yml | 54 + .../templates/configuration-qemu.toml.j2 | 706 +++ .../templates/containerd-shim-kata-v2.j2 | 2 + .../roles/container-engine/meta/main.yml | 58 + .../nerdctl/handlers/main.yml | 12 + .../container-engine/nerdctl/tasks/main.yml | 36 + .../nerdctl/templates/nerdctl.toml.j2 | 9 + .../container-engine/runc/defaults/main.yml | 5 + .../container-engine/runc/tasks/main.yml | 38 + .../container-engine/skopeo/tasks/main.yml | 32 + .../validate-container-engine/tasks/main.yml | 153 + .../container-engine/youki/defaults/main.yml | 3 + .../youki/molecule/default/converge.yml | 11 + .../molecule/default/files/10-mynet.conf | 17 + .../molecule/default/files/container.json | 10 + .../youki/molecule/default/files/sandbox.json | 10 + .../youki/molecule/default/molecule.yml | 39 + .../youki/molecule/default/prepare.yml | 49 + .../molecule/default/tests/test_default.py | 29 + .../container-engine/youki/tasks/main.yml | 12 + .../download/defaults/main/checksums.yml | 1226 +++++ .../roles/download/defaults/main/main.yml | 1173 +++++ kubespray/roles/download/meta/main.yml | 2 + .../download/tasks/check_pull_required.yml | 25 + .../download/tasks/download_container.yml | 125 + .../roles/download/tasks/download_file.yml | 145 + .../roles/download/tasks/extract_file.yml | 11 + kubespray/roles/download/tasks/main.yml | 30 + .../roles/download/tasks/prep_download.yml | 92 + .../download/tasks/prep_kubeadm_images.yml | 71 + .../download/tasks/set_container_facts.yml | 55 + .../download/templates/kubeadm-images.yaml.j2 | 25 + kubespray/roles/etcd/defaults/main.yml | 122 + kubespray/roles/etcd/handlers/backup.yml | 63 + .../roles/etcd/handlers/backup_cleanup.yml | 12 + kubespray/roles/etcd/handlers/main.yml | 64 + kubespray/roles/etcd/meta/main.yml | 8 + kubespray/roles/etcd/tasks/check_certs.yml | 171 + kubespray/roles/etcd/tasks/configure.yml | 173 + .../roles/etcd/tasks/gen_certs_script.yml | 171 + .../etcd/tasks/gen_nodes_certs_script.yml | 31 + kubespray/roles/etcd/tasks/install_docker.yml | 42 + kubespray/roles/etcd/tasks/install_host.yml | 31 + .../etcd/tasks/join_etcd-events_member.yml | 49 + .../roles/etcd/tasks/join_etcd_member.yml | 53 + kubespray/roles/etcd/tasks/main.yml | 96 + kubespray/roles/etcd/tasks/refresh_config.yml | 16 + kubespray/roles/etcd/tasks/upd_ca_trust.yml | 37 + .../etcd/templates/etcd-docker.service.j2 | 18 + .../templates/etcd-events-docker.service.j2 | 18 + .../templates/etcd-events-host.service.j2 | 16 + .../roles/etcd/templates/etcd-events.env.j2 | 43 + kubespray/roles/etcd/templates/etcd-events.j2 | 21 + .../roles/etcd/templates/etcd-host.service.j2 | 16 + kubespray/roles/etcd/templates/etcd.env.j2 | 70 + kubespray/roles/etcd/templates/etcd.j2 | 21 + .../roles/etcd/templates/make-ssl-etcd.sh.j2 | 105 + .../roles/etcd/templates/openssl.conf.j2 | 45 + .../roles/etcdctl_etcdutl/tasks/main.yml | 45 + .../etcdctl_etcdutl/templates/etcdctl.sh.j2 | 14 + kubespray/roles/helm-apps/README.md | 39 + .../roles/helm-apps/meta/argument_specs.yml | 93 + kubespray/roles/helm-apps/meta/main.yml | 3 + kubespray/roles/helm-apps/tasks/main.yml | 19 + kubespray/roles/helm-apps/vars/main.yml | 9 + .../kubernetes-apps/ansible/defaults/main.yml | 111 + .../ansible/tasks/cleanup_dns.yml | 44 + .../kubernetes-apps/ansible/tasks/coredns.yml | 44 + .../ansible/tasks/dashboard.yml | 21 + .../ansible/tasks/etcd_metrics.yml | 22 + .../kubernetes-apps/ansible/tasks/main.yml | 82 + .../ansible/tasks/netchecker.yml | 56 + .../ansible/tasks/nodelocaldns.yml | 79 + .../templates/coredns-clusterrole.yml.j2 | 32 + .../coredns-clusterrolebinding.yml.j2 | 18 + .../ansible/templates/coredns-config.yml.j2 | 85 + .../templates/coredns-deployment.yml.j2 | 124 + .../ansible/templates/coredns-sa.yml.j2 | 9 + .../ansible/templates/coredns-svc.yml.j2 | 28 + .../ansible/templates/dashboard.yml.j2 | 339 ++ .../dns-autoscaler-clusterrole.yml.j2 | 34 + .../dns-autoscaler-clusterrolebinding.yml.j2 | 29 + .../templates/dns-autoscaler-sa.yml.j2 | 22 + .../ansible/templates/dns-autoscaler.yml.j2 | 88 + .../templates/etcd_metrics-endpoints.yml.j2 | 20 + .../templates/etcd_metrics-service.yml.j2 | 13 + .../templates/netchecker-agent-ds.yml.j2 | 56 + ...etchecker-agent-hostnet-clusterrole.yml.j2 | 14 + ...er-agent-hostnet-clusterrolebinding.yml.j2 | 13 + .../netchecker-agent-hostnet-ds.yml.j2 | 58 + .../netchecker-agent-hostnet-psp.yml.j2 | 44 + .../templates/netchecker-agent-sa.yml.j2 | 5 + .../ansible/templates/netchecker-ns.yml.j2 | 6 + .../netchecker-server-clusterrole.yml.j2 | 9 + ...etchecker-server-clusterrolebinding.yml.j2 | 13 + .../netchecker-server-deployment.yml.j2 | 83 + .../templates/netchecker-server-sa.yml.j2 | 5 + .../templates/netchecker-server-svc.yml.j2 | 15 + .../templates/nodelocaldns-config.yml.j2 | 182 + .../templates/nodelocaldns-daemonset.yml.j2 | 115 + .../ansible/templates/nodelocaldns-sa.yml.j2 | 7 + .../nodelocaldns-second-daemonset.yml.j2 | 103 + .../kubernetes-apps/argocd/defaults/main.yml | 6 + .../kubernetes-apps/argocd/tasks/main.yml | 107 + .../argocd/templates/argocd-namespace.yml.j2 | 7 + .../cloud_controller/oci/defaults/main.yml | 6 + .../oci/tasks/credentials-check.yml | 67 + .../cloud_controller/oci/tasks/main.yml | 35 + .../controller-manager-config.yml.j2 | 89 + .../oci/templates/oci-cloud-provider.yml.j2 | 72 + .../cluster_roles/defaults/main.yml | 65 + .../files/k8s-cluster-critical-pc.yml | 8 + .../cluster_roles/files/oci-rbac.yml | 124 + .../cluster_roles/tasks/main.yml | 87 + .../cluster_roles/tasks/oci.yml | 19 + .../cluster_roles/templates/namespace.j2 | 4 + .../cluster_roles/templates/node-crb.yml.j2 | 17 + .../templates/vsphere-rbac.yml.j2 | 35 + .../meta/main.yml | 8 + .../nvidia_gpu/defaults/main.yml | 14 + .../nvidia_gpu/tasks/main.yml | 55 + .../k8s-device-plugin-nvidia-daemonset.yml.j2 | 60 + .../nvidia-driver-install-daemonset.yml.j2 | 82 + .../nvidia_gpu/vars/centos-7.yml | 3 + .../nvidia_gpu/vars/ubuntu-16.yml | 3 + .../nvidia_gpu/vars/ubuntu-18.yml | 3 + .../container_runtimes/crun/tasks/main.yaml | 19 + .../crun/templates/runtimeclass-crun.yml | 6 + .../container_runtimes/gvisor/tasks/main.yaml | 34 + .../templates/runtimeclass-gvisor.yml.j2 | 6 + .../kata_containers/defaults/main.yaml | 5 + .../kata_containers/tasks/main.yaml | 35 + .../templates/runtimeclass-kata-qemu.yml.j2 | 12 + .../container_runtimes/meta/main.yml | 31 + .../container_runtimes/youki/tasks/main.yaml | 19 + .../youki/templates/runtimeclass-youki.yml | 6 + .../roles/kubernetes-apps/csi_driver/OWNERS | 6 + .../csi_driver/aws_ebs/defaults/main.yml | 11 + .../csi_driver/aws_ebs/tasks/main.yml | 26 + .../aws-ebs-csi-controllerservice-rbac.yml.j2 | 180 + .../aws-ebs-csi-controllerservice.yml.j2 | 131 + .../templates/aws-ebs-csi-driver.yml.j2 | 8 + .../templates/aws-ebs-csi-nodeservice.yml.j2 | 101 + .../csi_driver/azuredisk/defaults/main.yml | 6 + .../tasks/azure-credential-check.yml | 54 + .../csi_driver/azuredisk/tasks/main.yml | 45 + ...azure-csi-azuredisk-controller-rbac.yml.j2 | 230 + .../azure-csi-azuredisk-controller.yml.j2 | 179 + .../azure-csi-azuredisk-driver.yml.j2 | 10 + .../azure-csi-azuredisk-node-rbac.yml.j2 | 30 + .../templates/azure-csi-azuredisk-node.yml.j2 | 168 + .../azure-csi-cloud-config-secret.yml.j2 | 7 + .../templates/azure-csi-cloud-config.j2 | 14 + .../csi_driver/cinder/defaults/main.yml | 30 + .../cinder/tasks/cinder-credential-check.yml | 59 + .../cinder/tasks/cinder-write-cacert.yml | 11 + .../csi_driver/cinder/tasks/main.yml | 57 + .../cinder-csi-cloud-config-secret.yml.j2 | 10 + .../templates/cinder-csi-cloud-config.j2 | 44 + .../cinder-csi-controllerplugin-rbac.yml.j2 | 179 + .../cinder-csi-controllerplugin.yml.j2 | 171 + .../cinder/templates/cinder-csi-driver.yml.j2 | 10 + .../cinder-csi-nodeplugin-rbac.yml.j2 | 38 + .../templates/cinder-csi-nodeplugin.yml.j2 | 145 + .../cinder-csi-poddisruptionbudget.yml.j2 | 14 + .../csi_driver/csi_crd/tasks/main.yml | 26 + .../templates/volumesnapshotclasses.yml.j2 | 116 + .../templates/volumesnapshotcontents.yml.j2 | 305 ++ .../csi_crd/templates/volumesnapshots.yml.j2 | 231 + .../csi_driver/gcp_pd/defaults/main.yml | 2 + .../csi_driver/gcp_pd/tasks/main.yml | 47 + .../templates/gcp-pd-csi-controller.yml.j2 | 165 + .../templates/gcp-pd-csi-cred-secret.yml.j2 | 8 + .../gcp_pd/templates/gcp-pd-csi-node.yml.j2 | 112 + .../templates/gcp-pd-csi-sc-regional.yml.j2 | 9 + .../templates/gcp-pd-csi-sc-zonal.yml.j2 | 8 + .../gcp_pd/templates/gcp-pd-csi-setup.yml.j2 | 291 ++ .../csi_driver/upcloud/defaults/main.yml | 16 + .../csi_driver/upcloud/tasks/main.yml | 40 + .../templates/upcloud-csi-controller.yml.j2 | 93 + .../templates/upcloud-csi-cred-secret.yml.j2 | 9 + .../templates/upcloud-csi-driver.yml.j2 | 8 + .../upcloud/templates/upcloud-csi-node.yml.j2 | 101 + .../templates/upcloud-csi-setup.yml.j2 | 185 + .../csi_driver/vsphere/defaults/main.yml | 54 + .../csi_driver/vsphere/tasks/main.yml | 55 + .../tasks/vsphere-credentials-check.yml | 38 + .../templates/vsphere-csi-cloud-config.j2 | 9 + .../vsphere-csi-controller-config.yml.j2 | 31 + .../vsphere-csi-controller-deployment.yml.j2 | 257 + .../vsphere-csi-controller-rbac.yml.j2 | 89 + .../vsphere-csi-controller-service.yml.j2 | 19 + .../templates/vsphere-csi-driver.yml.j2 | 7 + .../templates/vsphere-csi-namespace.yml.j2 | 4 + .../templates/vsphere-csi-node-rbac.yml.j2 | 55 + .../vsphere/templates/vsphere-csi-node.yml.j2 | 170 + .../hcloud/defaults/main.yml | 14 + .../hcloud/tasks/main.yml | 30 + ...controller-manager-ds-with-networks.yml.j2 | 96 + ...-hcloud-cloud-controller-manager-ds.yml.j2 | 94 + ...external-hcloud-cloud-role-bindings.yml.j2 | 13 + .../external-hcloud-cloud-secret.yml.j2 | 15 + ...ternal-hcloud-cloud-service-account.yml.j2 | 6 + .../huaweicloud/defaults/main.yml | 19 + .../tasks/huaweicloud-credential-check.yml | 33 + .../huaweicloud/tasks/main.yml | 49 + ...external-huawei-cloud-config-secret.yml.j2 | 10 + .../templates/external-huawei-cloud-config.j2 | 23 + ...-huawei-cloud-controller-manager-ds.yml.j2 | 93 + ...ud-controller-manager-role-bindings.yml.j2 | 16 + ...awei-cloud-controller-manager-roles.yml.j2 | 117 + .../external_cloud_controller/meta/main.yml | 42 + .../openstack/OWNERS | 6 + .../openstack/defaults/main.yml | 25 + .../openstack/tasks/main.yml | 49 + .../tasks/openstack-credential-check.yml | 66 + ...ernal-openstack-cloud-config-secret.yml.j2 | 13 + .../external-openstack-cloud-config.j2 | 92 + ...enstack-cloud-controller-manager-ds.yml.j2 | 111 + ...ud-controller-manager-role-bindings.yml.j2 | 16 + ...tack-cloud-controller-manager-roles.yml.j2 | 109 + .../vsphere/defaults/main.yml | 14 + .../vsphere/tasks/main.yml | 49 + .../tasks/vsphere-credentials-check.yml | 32 + ...vsphere-cloud-controller-manager-ds.yml.j2 | 76 + ...ud-controller-manager-role-bindings.yml.j2 | 35 + ...here-cloud-controller-manager-roles.yml.j2 | 91 + ...nal-vsphere-cpi-cloud-config-secret.yml.j2 | 11 + .../external-vsphere-cpi-cloud-config.j2 | 8 + .../cephfs_provisioner/defaults/main.yml | 10 + .../cephfs_provisioner/tasks/main.yml | 80 + .../templates/00-namespace.yml.j2 | 7 + .../clusterrole-cephfs-provisioner.yml.j2 | 26 + ...usterrolebinding-cephfs-provisioner.yml.j2 | 13 + .../deploy-cephfs-provisioner.yml.j2 | 34 + .../templates/psp-cephfs-provisioner.yml.j2 | 44 + .../templates/role-cephfs-provisioner.yml.j2 | 13 + .../rolebinding-cephfs-provisioner.yml.j2 | 14 + .../templates/sa-cephfs-provisioner.yml.j2 | 6 + .../templates/sc-cephfs-provisioner.yml.j2 | 15 + .../secret-cephfs-provisioner.yml.j2 | 9 + .../local_path_provisioner/defaults/main.yml | 10 + .../local_path_provisioner/tasks/main.yml | 47 + ...cal-path-storage-clusterrolebinding.yml.j2 | 13 + .../templates/local-path-storage-cm.yml.j2 | 35 + .../templates/local-path-storage-cr.yml.j2 | 18 + .../local-path-storage-deployment.yml.j2 | 41 + .../templates/local-path-storage-ns.yml.j2 | 5 + .../templates/local-path-storage-sa.yml.j2 | 6 + .../templates/local-path-storage-sc.yml.j2 | 10 + .../defaults/main.yml | 20 + .../tasks/basedirs.yml | 12 + .../local_volume_provisioner/tasks/main.yml | 48 + ...ocal-volume-provisioner-clusterrole.yml.j2 | 22 + ...lume-provisioner-clusterrolebinding.yml.j2 | 14 + .../local-volume-provisioner-cm.yml.j2 | 33 + .../local-volume-provisioner-ds.yml.j2 | 66 + .../local-volume-provisioner-ns.yml.j2 | 7 + .../local-volume-provisioner-sa.yml.j2 | 6 + .../local-volume-provisioner-sc.yml.j2 | 12 + .../external_provisioner/meta/main.yml | 30 + .../rbd_provisioner/defaults/main.yml | 17 + .../rbd_provisioner/tasks/main.yml | 80 + .../templates/00-namespace.yml.j2 | 7 + .../clusterrole-rbd-provisioner.yml.j2 | 30 + .../clusterrolebinding-rbd-provisioner.yml.j2 | 13 + .../templates/deploy-rbd-provisioner.yml.j2 | 40 + .../templates/psp-rbd-provisioner.yml.j2 | 44 + .../templates/role-rbd-provisioner.yml.j2 | 13 + .../rolebinding-rbd-provisioner.yml.j2 | 14 + .../templates/sa-rbd-provisioner.yml.j2 | 6 + .../templates/sc-rbd-provisioner.yml.j2 | 19 + .../templates/secret-rbd-provisioner.yml.j2 | 18 + kubespray/roles/kubernetes-apps/helm/.gitkeep | 0 .../kubernetes-apps/helm/defaults/main.yml | 2 + .../roles/kubernetes-apps/helm/tasks/main.yml | 49 + .../helm/tasks/pyyaml-flatcar.yml | 22 + .../kubernetes-apps/helm/vars/amazon.yml | 2 + .../kubernetes-apps/helm/vars/centos-7.yml | 2 + .../kubernetes-apps/helm/vars/centos.yml | 2 + .../kubernetes-apps/helm/vars/debian.yml | 2 + .../kubernetes-apps/helm/vars/fedora.yml | 2 + .../kubernetes-apps/helm/vars/redhat-7.yml | 2 + .../kubernetes-apps/helm/vars/redhat.yml | 2 + .../roles/kubernetes-apps/helm/vars/suse.yml | 2 + .../kubernetes-apps/helm/vars/ubuntu.yml | 2 + .../alb_ingress_controller/OWNERS | 6 + .../alb_ingress_controller/defaults/main.yml | 7 + .../alb_ingress_controller/tasks/main.yml | 36 + .../templates/alb-ingress-clusterrole.yml.j2 | 13 + .../alb-ingress-clusterrolebinding.yml.j2 | 14 + .../templates/alb-ingress-deploy.yml.j2 | 74 + .../templates/alb-ingress-ns.yml.j2 | 7 + .../templates/alb-ingress-sa.yml.j2 | 6 + .../cert_manager/defaults/main.yml | 19 + .../cert_manager/tasks/main.yml | 56 + .../templates/cert-manager.crds.yml.j2 | 4422 +++++++++++++++++ .../templates/cert-manager.yml.j2 | 1224 +++++ .../ingress_nginx/defaults/main.yml | 22 + .../ingress_nginx/tasks/main.yml | 61 + .../templates/00-namespace.yml.j2 | 7 + .../admission-webhook-configuration.yml.j2 | 29 + .../templates/admission-webhook-job.yml.j2 | 86 + .../clusterrole-admission-webhook.yml.j2 | 15 + .../clusterrole-ingress-nginx.yml.j2 | 36 + ...lusterrolebinding-admission-webhook.yml.j2 | 16 + .../clusterrolebinding-ingress-nginx.yml.j2 | 16 + .../templates/cm-ingress-nginx.yml.j2 | 13 + .../templates/cm-tcp-services.yml.j2 | 13 + .../templates/cm-udp-services.yml.j2 | 13 + .../ds-ingress-nginx-controller.yml.j2 | 140 + .../templates/ingressclass-nginx.yml.j2 | 13 + .../templates/role-admission-webhook.yml.j2 | 17 + .../templates/role-ingress-nginx.yml.j2 | 53 + .../rolebinding-admission-webhook.yml.j2 | 17 + .../rolebinding-ingress-nginx.yml.j2 | 17 + .../templates/sa-admission-webhook.yml.j2 | 8 + .../templates/sa-ingress-nginx.yml.j2 | 9 + .../ingress_controller/meta/main.yml | 22 + .../kubernetes-apps/krew/defaults/main.yml | 5 + .../roles/kubernetes-apps/krew/tasks/krew.yml | 38 + .../roles/kubernetes-apps/krew/tasks/main.yml | 10 + .../kubernetes-apps/krew/templates/krew.j2 | 7 + .../krew/templates/krew.yml.j2 | 100 + .../kubelet-csr-approver/defaults/main.yml | 12 + .../kubelet-csr-approver/meta/main.yml | 20 + kubespray/roles/kubernetes-apps/meta/main.yml | 126 + .../roles/kubernetes-apps/metallb/OWNERS | 5 + .../kubernetes-apps/metallb/defaults/main.yml | 18 + .../kubernetes-apps/metallb/tasks/main.yml | 123 + .../metallb/templates/layer2.yaml.j2 | 19 + .../metallb/templates/layer3.yaml.j2 | 125 + .../metallb/templates/metallb.yaml.j2 | 2035 ++++++++ .../metallb/templates/pools.yaml.j2 | 22 + .../metrics_server/defaults/main.yml | 14 + .../metrics_server/tasks/main.yml | 57 + .../templates/auth-delegator.yaml.j2 | 14 + .../templates/auth-reader.yaml.j2 | 15 + .../templates/metrics-apiservice.yaml.j2 | 15 + .../metrics-server-deployment.yaml.j2 | 121 + .../templates/metrics-server-sa.yaml.j2 | 8 + .../templates/metrics-server-service.yaml.j2 | 17 + ...resource-reader-clusterrolebinding.yaml.j2 | 15 + .../templates/resource-reader.yaml.j2 | 22 + .../network_plugin/calico/tasks/main.yml | 2 + .../network_plugin/flannel/tasks/main.yml | 17 + .../network_plugin/kube-ovn/tasks/main.yml | 9 + .../network_plugin/kube-router/OWNERS | 6 + .../network_plugin/kube-router/tasks/main.yml | 23 + .../network_plugin/meta/main.yml | 31 + .../network_plugin/multus/tasks/main.yml | 18 + .../network_plugin/weave/tasks/main.yml | 21 + .../persistent_volumes/aws-ebs-csi/OWNERS | 5 + .../aws-ebs-csi/defaults/main.yml | 8 + .../aws-ebs-csi/tasks/main.yml | 20 + .../aws-ebs-csi-storage-class.yml.j2 | 18 + .../azuredisk-csi/defaults/main.yml | 3 + .../azuredisk-csi/tasks/main.yml | 20 + .../templates/azure-csi-storage-class.yml.j2 | 14 + .../cinder-csi/defaults/main.yml | 7 + .../cinder-csi/tasks/main.yml | 20 + .../templates/cinder-csi-storage-class.yml.j2 | 25 + .../gcp-pd-csi/defaults/main.yml | 8 + .../gcp-pd-csi/tasks/main.yml | 20 + .../templates/gcp-pd-csi-storage-class.yml.j2 | 20 + .../persistent_volumes/meta/main.yml | 43 + .../openstack/defaults/main.yml | 7 + .../openstack/tasks/main.yml | 20 + .../templates/openstack-storage-class.yml.j2 | 15 + .../upcloud-csi/defaults/main.yml | 12 + .../upcloud-csi/tasks/main.yml | 20 + .../upcloud-csi-storage-class.yml.j2 | 16 + .../calico/defaults/main.yml | 10 + .../policy_controller/calico/tasks/main.yml | 34 + .../templates/calico-kube-controllers.yml.j2 | 87 + .../calico/templates/calico-kube-cr.yml.j2 | 110 + .../calico/templates/calico-kube-crb.yml.j2 | 13 + .../calico/templates/calico-kube-sa.yml.j2 | 6 + .../policy_controller/meta/main.yml | 8 + .../registry/defaults/main.yml | 48 + .../kubernetes-apps/registry/tasks/main.yml | 109 + .../registry/templates/registry-cm.yml.j2 | 10 + .../registry/templates/registry-cr.yml.j2 | 15 + .../registry/templates/registry-crb.yml.j2 | 13 + .../registry/templates/registry-ing.yml.j2 | 27 + .../registry/templates/registry-ns.yml.j2 | 7 + .../registry/templates/registry-psp.yml.j2 | 44 + .../registry/templates/registry-pvc.yml.j2 | 15 + .../registry/templates/registry-rs.yml.j2 | 115 + .../registry/templates/registry-sa.yml.j2 | 5 + .../templates/registry-secrets.yml.j2 | 10 + .../registry/templates/registry-svc.yml.j2 | 32 + .../snapshots/cinder-csi/defaults/main.yml | 6 + .../snapshots/cinder-csi/tasks/main.yml | 18 + .../cinder-csi-snapshot-class.yml.j2 | 13 + .../kubernetes-apps/snapshots/meta/main.yml | 14 + .../snapshot-controller/defaults/main.yml | 3 + .../snapshot-controller/tasks/main.yml | 39 + .../templates/rbac-snapshot-controller.yml.j2 | 85 + .../templates/snapshot-controller.yml.j2 | 40 + .../templates/snapshot-ns.yml.j2 | 7 + .../roles/kubernetes/client/defaults/main.yml | 8 + .../roles/kubernetes/client/tasks/main.yml | 114 + .../control-plane/defaults/main/etcd.yml | 31 + .../defaults/main/kube-proxy.yml | 114 + .../defaults/main/kube-scheduler.yml | 33 + .../control-plane/defaults/main/main.yml | 230 + .../control-plane/handlers/main.yml | 135 + .../kubernetes/control-plane/meta/main.yml | 11 + .../tasks/define-first-kube-control.yml | 19 + .../control-plane/tasks/encrypt-at-rest.yml | 40 + .../control-plane/tasks/kubeadm-backup.yml | 28 + .../control-plane/tasks/kubeadm-etcd.yml | 29 + .../tasks/kubeadm-fix-apiserver.yml | 24 + .../control-plane/tasks/kubeadm-secondary.yml | 80 + .../control-plane/tasks/kubeadm-setup.yml | 249 + .../control-plane/tasks/kubeadm-upgrade.yml | 78 + .../kubelet-fix-client-cert-rotation.yml | 18 + .../kubernetes/control-plane/tasks/main.yml | 108 + .../control-plane/tasks/pre-upgrade.yml | 21 + .../control-plane/tasks/psp-install.yml | 38 + .../templates/admission-controls.yaml.j2 | 9 + .../templates/apiserver-audit-policy.yaml.j2 | 129 + .../apiserver-audit-webhook-config.yaml.j2 | 17 + .../templates/eventratelimit.yaml.j2 | 11 + .../templates/k8s-certs-renew.service.j2 | 6 + .../templates/k8s-certs-renew.sh.j2 | 23 + .../templates/k8s-certs-renew.timer.j2 | 8 + .../templates/kubeadm-config.v1beta3.yaml.j2 | 456 ++ .../kubeadm-controlplane.v1beta3.yaml.j2 | 34 + .../templates/kubescheduler-config.yaml.j2 | 24 + .../templates/podsecurity.yaml.j2 | 17 + .../control-plane/templates/psp-cr.yml.j2 | 32 + .../control-plane/templates/psp-crb.yml.j2 | 54 + .../control-plane/templates/psp.yml.j2 | 27 + .../templates/secrets_encryption.yaml.j2 | 11 + .../webhook-authorization-config.yaml.j2 | 18 + .../webhook-token-auth-config.yaml.j2 | 21 + .../kubernetes/control-plane/vars/main.yaml | 3 + .../kubernetes/kubeadm/defaults/main.yml | 12 + .../kubernetes/kubeadm/handlers/main.yml | 15 + .../kubeadm/tasks/kubeadm_etcd_node.yml | 62 + .../roles/kubernetes/kubeadm/tasks/main.yml | 177 + .../templates/kubeadm-client.conf.v1beta3.j2 | 39 + .../kubernetes/node-label/tasks/main.yml | 49 + .../roles/kubernetes/node/defaults/main.yml | 242 + .../roles/kubernetes/node/handlers/main.yml | 15 + .../azure-credential-check.yml | 82 + .../openstack-credential-check.yml | 34 + .../vsphere-credential-check.yml | 22 + .../roles/kubernetes/node/tasks/facts.yml | 62 + .../roles/kubernetes/node/tasks/install.yml | 22 + .../roles/kubernetes/node/tasks/kubelet.yml | 52 + .../node/tasks/loadbalancer/haproxy.yml | 34 + .../node/tasks/loadbalancer/kube-vip.yml | 13 + .../node/tasks/loadbalancer/nginx-proxy.yml | 34 + .../roles/kubernetes/node/tasks/main.yml | 202 + .../kubernetes/node/tasks/pre_upgrade.yml | 48 + .../cloud-configs/aws-cloud-config.j2 | 11 + .../cloud-configs/azure-cloud-config.j2 | 26 + .../cloud-configs/gce-cloud-config.j2 | 2 + .../cloud-configs/openstack-cloud-config.j2 | 54 + .../cloud-configs/vsphere-cloud-config.j2 | 36 + .../node/templates/http-proxy.conf.j2 | 2 + .../templates/kubelet-config.v1beta1.yaml.j2 | 170 + .../node/templates/kubelet.env.v1beta1.j2 | 43 + .../node/templates/kubelet.service.j2 | 52 + .../templates/loadbalancer/haproxy.cfg.j2 | 49 + .../node/templates/loadbalancer/nginx.conf.j2 | 60 + .../templates/manifests/haproxy.manifest.j2 | 42 + .../templates/manifests/kube-vip.manifest.j2 | 102 + .../manifests/nginx-proxy.manifest.j2 | 42 + .../node/templates/node-kubeconfig.yaml.j2 | 19 + .../roles/kubernetes/node/vars/fedora.yml | 2 + .../roles/kubernetes/node/vars/ubuntu-18.yml | 2 + .../roles/kubernetes/node/vars/ubuntu-20.yml | 2 + .../roles/kubernetes/node/vars/ubuntu-22.yml | 2 + .../kubernetes/preinstall/defaults/main.yml | 152 + .../preinstall/files/dhclient_nodnsupdate | 4 + .../kubernetes/preinstall/gen-gitinfos.sh | 73 + .../kubernetes/preinstall/handlers/main.yml | 146 + .../roles/kubernetes/preinstall/meta/main.yml | 8 + .../preinstall/tasks/0010-swapoff.yml | 28 + .../preinstall/tasks/0020-set_facts.yml | 270 + .../preinstall/tasks/0040-verify-settings.yml | 318 ++ .../tasks/0050-create_directories.yml | 119 + .../preinstall/tasks/0060-resolvconf.yml | 58 + .../tasks/0061-systemd-resolved.yml | 9 + .../0062-networkmanager-unmanaged-devices.yml | 28 + .../tasks/0063-networkmanager-dns.yml | 37 + .../preinstall/tasks/0070-system-packages.yml | 99 + .../tasks/0080-system-configurations.yml | 146 + .../tasks/0081-ntp-configurations.yml | 87 + .../preinstall/tasks/0090-etchosts.yml | 81 + .../preinstall/tasks/0100-dhclient-hooks.yml | 33 + .../tasks/0110-dhclient-hooks-undo.yml | 18 + .../tasks/0120-growpart-azure-centos-7.yml | 44 + .../kubernetes/preinstall/tasks/main.yml | 148 + .../preinstall/templates/ansible_git.j2 | 3 + .../preinstall/templates/chrony.conf.j2 | 27 + .../templates/dhclient_dnsupdate.sh.j2 | 13 + .../templates/dhclient_dnsupdate_rh.sh.j2 | 17 + .../preinstall/templates/ntp.conf.j2 | 45 + .../preinstall/templates/resolvconf.j2 | 10 + .../preinstall/templates/resolved.conf.j2 | 21 + .../kubernetes/preinstall/vars/amazon.yml | 7 + .../kubernetes/preinstall/vars/centos.yml | 8 + .../kubernetes/preinstall/vars/debian-11.yml | 10 + .../kubernetes/preinstall/vars/debian-12.yml | 11 + .../kubernetes/preinstall/vars/debian.yml | 9 + .../kubernetes/preinstall/vars/fedora.yml | 8 + .../kubernetes/preinstall/vars/redhat.yml | 8 + .../roles/kubernetes/preinstall/vars/suse.yml | 5 + .../kubernetes/preinstall/vars/ubuntu.yml | 8 + .../kubernetes/tokens/files/kube-gen-token.sh | 34 + .../kubernetes/tokens/tasks/check-tokens.yml | 41 + .../kubernetes/tokens/tasks/gen_tokens.yml | 63 + .../roles/kubernetes/tokens/tasks/main.yml | 21 + .../kubespray-defaults/defaults/main.yaml | 681 +++ .../roles/kubespray-defaults/meta/main.yml | 6 + .../kubespray-defaults/tasks/fallback_ips.yml | 33 + .../roles/kubespray-defaults/tasks/main.yaml | 33 + .../kubespray-defaults/tasks/no_proxy.yml | 40 + .../roles/kubespray-defaults/vars/main.yml | 9 + kubespray/roles/moaroom/defaults/main.yml | 24 + .../moaroom/templates/deploy-control.yml.j2 | 102 + .../network_plugin/calico/defaults/main.yml | 171 + .../network_plugin/calico/files/openssl.conf | 27 + .../network_plugin/calico/handlers/main.yml | 31 + .../calico/rr/defaults/main.yml | 5 + .../network_plugin/calico/rr/tasks/main.yml | 16 + .../network_plugin/calico/rr/tasks/pre.yml | 15 + .../calico/rr/tasks/update-node.yml | 50 + .../calico/tasks/calico_apiserver_certs.yml | 60 + .../network_plugin/calico/tasks/check.yml | 207 + .../network_plugin/calico/tasks/install.yml | 479 ++ .../network_plugin/calico/tasks/main.yml | 9 + .../calico/tasks/peer_with_calico_rr.yml | 86 + .../calico/tasks/peer_with_router.yml | 77 + .../roles/network_plugin/calico/tasks/pre.yml | 47 + .../network_plugin/calico/tasks/repos.yml | 21 + .../network_plugin/calico/tasks/reset.yml | 30 + .../calico/tasks/typha_certs.yml | 51 + .../templates/calico-apiserver-ns.yml.j2 | 10 + .../calico/templates/calico-apiserver.yml.j2 | 290 ++ .../calico/templates/calico-config.yml.j2 | 111 + .../calico/templates/calico-cr.yml.j2 | 203 + .../calico/templates/calico-crb.yml.j2 | 28 + .../calico/templates/calico-ipamconfig.yml.j2 | 8 + .../calico/templates/calico-node-sa.yml.j2 | 13 + .../calico/templates/calico-node.yml.j2 | 467 ++ .../calico/templates/calico-typha.yml.j2 | 190 + .../calico/templates/calicoctl.etcd.sh.j2 | 6 + .../calico/templates/calicoctl.kdd.sh.j2 | 8 + .../kubernetes-services-endpoint.yml.j2 | 11 + .../calico/templates/make-ssl-calico.sh.j2 | 102 + .../network_plugin/calico/vars/amazon.yml | 5 + .../network_plugin/calico/vars/centos-9.yml | 3 + .../network_plugin/calico/vars/debian.yml | 3 + .../network_plugin/calico/vars/fedora.yml | 3 + .../network_plugin/calico/vars/opensuse.yml | 3 + .../network_plugin/calico/vars/redhat-9.yml | 3 + .../network_plugin/calico/vars/redhat.yml | 4 + .../network_plugin/calico/vars/rocky-9.yml | 3 + .../network_plugin/cilium/defaults/main.yml | 311 ++ .../network_plugin/cilium/tasks/apply.yml | 33 + .../network_plugin/cilium/tasks/check.yml | 63 + .../network_plugin/cilium/tasks/install.yml | 97 + .../network_plugin/cilium/tasks/main.yml | 9 + .../network_plugin/cilium/tasks/reset.yml | 9 + .../cilium/tasks/reset_iface.yml | 12 + .../templates/000-cilium-portmap.conflist.j2 | 13 + .../templates/cilium-operator/cr.yml.j2 | 169 + .../templates/cilium-operator/crb.yml.j2 | 13 + .../templates/cilium-operator/deploy.yml.j2 | 170 + .../templates/cilium-operator/sa.yml.j2 | 6 + .../cilium/templates/cilium/config.yml.j2 | 256 + .../cilium/templates/cilium/cr.yml.j2 | 122 + .../cilium/templates/cilium/crb.yml.j2 | 13 + .../cilium/templates/cilium/ds.yml.j2 | 444 ++ .../cilium/templates/cilium/sa.yml.j2 | 6 + .../cilium/templates/cilium/secret.yml.j2 | 9 + .../cilium/templates/hubble/config.yml.j2 | 70 + .../cilium/templates/hubble/cr.yml.j2 | 106 + .../cilium/templates/hubble/crb.yml.j2 | 44 + .../cilium/templates/hubble/cronjob.yml.j2 | 38 + .../cilium/templates/hubble/deploy.yml.j2 | 192 + .../cilium/templates/hubble/job.yml.j2 | 34 + .../cilium/templates/hubble/sa.yml.j2 | 23 + .../cilium/templates/hubble/service.yml.j2 | 102 + .../roles/network_plugin/cni/tasks/main.yml | 16 + .../custom_cni/defaults/main.yml | 3 + .../network_plugin/custom_cni/tasks/main.yml | 26 + .../network_plugin/flannel/defaults/main.yml | 28 + .../network_plugin/flannel/meta/main.yml | 3 + .../network_plugin/flannel/tasks/main.yml | 21 + .../network_plugin/flannel/tasks/reset.yml | 24 + .../flannel/templates/cni-flannel-rbac.yml.j2 | 52 + .../flannel/templates/cni-flannel.yml.j2 | 170 + .../roles/network_plugin/kube-ovn/OWNERS | 4 + .../network_plugin/kube-ovn/defaults/main.yml | 118 + .../network_plugin/kube-ovn/tasks/main.yml | 17 + .../templates/cni-kube-ovn-crd.yml.j2 | 1533 ++++++ .../kube-ovn/templates/cni-kube-ovn.yml.j2 | 673 +++ .../kube-ovn/templates/cni-ovn.yml.j2 | 517 ++ .../roles/network_plugin/kube-router/OWNERS | 6 + .../kube-router/defaults/main.yml | 66 + .../kube-router/handlers/main.yml | 24 + .../network_plugin/kube-router/meta/main.yml | 3 + .../kube-router/tasks/annotate.yml | 21 + .../network_plugin/kube-router/tasks/main.yml | 62 + .../kube-router/tasks/reset.yml | 28 + .../kube-router/templates/cni-conf.json.j2 | 27 + .../kube-router/templates/kube-router.yml.j2 | 220 + .../kube-router/templates/kubeconfig.yml.j2 | 18 + kubespray/roles/network_plugin/macvlan/OWNERS | 6 + .../network_plugin/macvlan/defaults/main.yml | 6 + .../network_plugin/macvlan/files/ifdown-local | 6 + .../macvlan/files/ifdown-macvlan | 40 + .../network_plugin/macvlan/files/ifup-local | 6 + .../network_plugin/macvlan/files/ifup-macvlan | 43 + .../network_plugin/macvlan/handlers/main.yml | 20 + .../network_plugin/macvlan/meta/main.yml | 3 + .../network_plugin/macvlan/tasks/main.yml | 110 + .../macvlan/templates/10-macvlan.conf.j2 | 15 + .../macvlan/templates/99-loopback.conf.j2 | 5 + .../templates/centos-network-macvlan.cfg.j2 | 13 + .../templates/centos-postdown-macvlan.cfg.j2 | 3 + .../templates/centos-postup-macvlan.cfg.j2 | 3 + .../templates/centos-routes-macvlan.cfg.j2 | 7 + .../templates/coreos-device-macvlan.cfg.j2 | 6 + .../templates/coreos-interface-macvlan.cfg.j2 | 6 + .../templates/coreos-network-macvlan.cfg.j2 | 17 + .../templates/coreos-service-nat_ouside.j2 | 6 + .../templates/debian-network-macvlan.cfg.j2 | 26 + kubespray/roles/network_plugin/meta/main.yml | 48 + .../network_plugin/multus/defaults/main.yml | 10 + .../multus/files/multus-clusterrole.yml | 28 + .../files/multus-clusterrolebinding.yml | 13 + .../multus/files/multus-crd.yml | 45 + .../multus/files/multus-serviceaccount.yml | 6 + .../roles/network_plugin/multus/meta/main.yml | 3 + .../network_plugin/multus/tasks/main.yml | 36 + .../multus/templates/multus-daemonset.yml.j2 | 79 + .../network_plugin/ovn4nfv/tasks/main.yml | 16 + .../network_plugin/weave/defaults/main.yml | 64 + .../roles/network_plugin/weave/meta/main.yml | 3 + .../roles/network_plugin/weave/tasks/main.yml | 12 + .../weave/templates/10-weave.conflist.j2 | 16 + .../weave/templates/weave-net.yml.j2 | 297 ++ kubespray/roles/recover_control_plane/OWNERS | 8 + .../control-plane/defaults/main.yml | 2 + .../control-plane/tasks/main.yml | 29 + .../recover_control_plane/etcd/tasks/main.yml | 93 + .../etcd/tasks/recover_lost_quorum.yml | 59 + .../post-recover/tasks/main.yml | 20 + .../remove-node/post-remove/defaults/main.yml | 3 + .../remove-node/post-remove/tasks/main.yml | 13 + .../remove-node/pre-remove/defaults/main.yml | 6 + .../remove-node/pre-remove/tasks/main.yml | 43 + .../remove-etcd-node/tasks/main.yml | 58 + kubespray/roles/reset/defaults/main.yml | 18 + kubespray/roles/reset/tasks/main.yml | 439 ++ .../upgrade/post-upgrade/defaults/main.yml | 5 + .../roles/upgrade/post-upgrade/tasks/main.yml | 32 + .../upgrade/pre-upgrade/defaults/main.yml | 20 + .../roles/upgrade/pre-upgrade/tasks/main.yml | 131 + .../upgrade/system-upgrade/tasks/apt.yml | 13 + .../upgrade/system-upgrade/tasks/main.yml | 17 + .../upgrade/system-upgrade/tasks/yum.yml | 12 + .../kubernetes_patch/defaults/main.yml | 4 + .../win_nodes/kubernetes_patch/tasks/main.yml | 41 + kubespray/run.rc | 46 + kubespray/scale.yml | 3 + kubespray/scripts/collect-info.yaml | 153 + kubespray/scripts/download_hash.py | 65 + kubespray/scripts/download_hash.sh | 259 + kubespray/scripts/gen_tags.sh | 12 + .../scripts/gitlab-branch-cleanup/.gitignore | 2 + .../scripts/gitlab-branch-cleanup/README.md | 24 + .../scripts/gitlab-branch-cleanup/main.py | 38 + .../gitlab-branch-cleanup/requirements.txt | 1 + kubespray/scripts/gitlab-runner.sh | 22 + .../scripts/openstack-cleanup/.gitignore | 1 + kubespray/scripts/openstack-cleanup/README.md | 21 + kubespray/scripts/openstack-cleanup/main.py | 98 + .../openstack-cleanup/requirements.txt | 2 + kubespray/scripts/premoderator.sh | 51 + kubespray/setup.cfg | 62 + kubespray/setup.py | 19 + kubespray/test-infra/image-builder/Makefile | 2 + kubespray/test-infra/image-builder/OWNERS | 8 + .../test-infra/image-builder/cluster.yml | 6 + kubespray/test-infra/image-builder/hosts.ini | 4 + .../roles/kubevirt-images/defaults/main.yml | 112 + .../roles/kubevirt-images/tasks/main.yml | 58 + .../kubevirt-images/templates/Dockerfile | 6 + .../test-infra/vagrant-docker/Dockerfile | 16 + kubespray/test-infra/vagrant-docker/README.md | 24 + kubespray/test-infra/vagrant-docker/build.sh | 13 + kubespray/tests/Makefile | 90 + kubespray/tests/README.md | 40 + kubespray/tests/ansible.cfg | 15 + .../tests/cloud_playbooks/cleanup-packet.yml | 8 + .../tests/cloud_playbooks/create-aws.yml | 26 + kubespray/tests/cloud_playbooks/create-do.yml | 94 + .../tests/cloud_playbooks/create-gce.yml | 81 + .../tests/cloud_playbooks/create-packet.yml | 11 + .../tests/cloud_playbooks/delete-aws.yml | 19 + .../tests/cloud_playbooks/delete-gce.yml | 50 + .../tests/cloud_playbooks/delete-packet.yml | 11 + .../roles/cleanup-packet-ci/tasks/main.yml | 16 + .../roles/packet-ci/defaults/main.yml | 44 + .../roles/packet-ci/tasks/cleanup-old-vms.yml | 17 + .../roles/packet-ci/tasks/create-vms.yml | 50 + .../roles/packet-ci/tasks/delete-vms.yml | 30 + .../roles/packet-ci/tasks/main.yml | 21 + .../roles/packet-ci/templates/inventory.j2 | 93 + .../roles/packet-ci/templates/vm.yml.j2 | 52 + .../tests/cloud_playbooks/templates/boto.j2 | 11 + .../templates/gcs_life.json.j2 | 9 + .../tests/cloud_playbooks/upload-logs-gcs.yml | 82 + .../tests/cloud_playbooks/wait-for-ssh.yml | 13 + .../common/_docker_hub_registry_mirror.yml | 45 + .../tests/common/_kubespray_test_settings.yml | 5 + kubespray/tests/files/custom_cni/README.md | 11 + kubespray/tests/files/custom_cni/cilium.yaml | 1056 ++++ kubespray/tests/files/custom_cni/values.yaml | 11 + .../packet_almalinux8-calico-ha-ebpf.yml | 10 + ...malinux8-calico-nodelocaldns-secondary.yml | 9 + .../packet_almalinux8-calico-remove-node.yml | 7 + .../tests/files/packet_almalinux8-calico.yml | 19 + .../tests/files/packet_almalinux8-crio.yml | 8 + .../tests/files/packet_almalinux8-docker.yml | 10 + .../files/packet_almalinux8-kube-ovn.yml | 8 + .../tests/files/packet_amazon-linux-2-aio.yml | 4 + ...acket_centos7-calico-ha-once-localhost.yml | 18 + .../tests/files/packet_centos7-calico-ha.yml | 13 + .../packet_centos7-flannel-addons-ha.yml | 74 + .../files/packet_centos7-multus-calico.yml | 7 + .../files/packet_centos7-weave-upgrade-ha.yml | 11 + .../tests/files/packet_debian10-calico.yml | 11 + .../packet_debian10-cilium-svc-proxy.yml | 10 + .../tests/files/packet_debian10-docker.yml | 9 + .../tests/files/packet_debian10-macvlan.yml | 11 + .../packet_debian11-calico-upgrade-once.yml | 16 + .../files/packet_debian11-calico-upgrade.yml | 13 + .../tests/files/packet_debian11-calico.yml | 4 + .../files/packet_debian11-custom-cni.yml | 9 + .../tests/files/packet_debian11-docker.yml | 9 + .../packet_debian11-kubelet-csr-approver.yml | 11 + .../tests/files/packet_debian12-calico.yml | 4 + .../tests/files/packet_debian12-cilium.yml | 7 + .../tests/files/packet_debian12-docker.yml | 9 + .../files/packet_fedora37-calico-selinux.yml | 14 + .../packet_fedora37-calico-swap-selinux.yml | 19 + .../tests/files/packet_fedora37-crio.yml | 15 + .../files/packet_fedora38-docker-calico.yml | 15 + .../files/packet_fedora38-docker-weave.yml | 12 + .../tests/files/packet_fedora38-kube-ovn.yml | 7 + .../files/packet_opensuse-docker-cilium.yml | 11 + .../tests/files/packet_rockylinux8-calico.yml | 11 + .../tests/files/packet_rockylinux9-calico.yml | 11 + .../tests/files/packet_rockylinux9-cilium.yml | 10 + .../files/packet_ubuntu20-aio-docker.yml | 16 + ...acket_ubuntu20-calico-aio-ansible-2_11.yml | 1 + .../packet_ubuntu20-calico-aio-hardening.yml | 106 + .../files/packet_ubuntu20-calico-aio.yml | 11 + ...buntu20-calico-etcd-kubeadm-upgrade-ha.yml | 24 + .../packet_ubuntu20-calico-etcd-kubeadm.yml | 11 + ...et_ubuntu20-calico-ha-recover-noquorum.yml | 4 + .../packet_ubuntu20-calico-ha-recover.yml | 4 + .../packet_ubuntu20-calico-ha-wireguard.yml | 13 + .../files/packet_ubuntu20-cilium-sep.yml | 9 + .../tests/files/packet_ubuntu20-crio.yml | 10 + .../packet_ubuntu20-docker-weave-sep.yml | 16 + .../files/packet_ubuntu20-flannel-ha-once.yml | 22 + .../files/packet_ubuntu20-flannel-ha.yml | 10 + .../files/packet_ubuntu22-aio-docker.yml | 17 + .../files/packet_ubuntu22-calico-aio.yml | 27 + .../tests/files/tf-elastx_ubuntu20-calico.yml | 5 + .../tests/files/tf-ovh_ubuntu20-calico.yml | 7 + .../files/vagrant_centos7-kube-router.rb | 15 + .../files/vagrant_centos7-kube-router.yml | 8 + .../files/vagrant_fedora37-kube-router.rb | 15 + .../files/vagrant_fedora37-kube-router.yml | 7 + .../vagrant_ubuntu20-calico-dual-stack.rb | 7 + .../vagrant_ubuntu20-calico-dual-stack.yml | 3 + .../vagrant_ubuntu20-flannel-collection.rb | 9 + .../vagrant_ubuntu20-flannel-collection.yml | 3 + .../tests/files/vagrant_ubuntu20-flannel.rb | 9 + .../tests/files/vagrant_ubuntu20-flannel.yml | 3 + .../files/vagrant_ubuntu20-kube-router-sep.rb | 15 + .../vagrant_ubuntu20-kube-router-sep.yml | 8 + .../vagrant_ubuntu20-kube-router-svc-proxy.rb | 10 + ...vagrant_ubuntu20-kube-router-svc-proxy.yml | 10 + .../files/vagrant_ubuntu20-weave-medium.rb | 7 + .../files/vagrant_ubuntu20-weave-medium.yml | 3 + .../tests/local_inventory/host_vars/localhost | 12 + kubespray/tests/local_inventory/hosts.cfg | 1 + kubespray/tests/requirements.txt | 11 + kubespray/tests/requirements.yml | 4 + kubespray/tests/run-tests.sh | 8 + kubespray/tests/scripts/ansibl8s_test.sh | 52 + .../tests/scripts/check_galaxy_version.sh | 18 + .../tests/scripts/check_readme_versions.sh | 33 + kubespray/tests/scripts/check_typo.sh | 12 + kubespray/tests/scripts/create-tf.sh | 5 + kubespray/tests/scripts/delete-tf.sh | 5 + kubespray/tests/scripts/md-table/main.py | 96 + .../tests/scripts/md-table/requirements.txt | 4 + kubespray/tests/scripts/md-table/table.md.j2 | 15 + kubespray/tests/scripts/md-table/test.sh | 11 + kubespray/tests/scripts/molecule_logs.sh | 9 + kubespray/tests/scripts/molecule_run.sh | 34 + kubespray/tests/scripts/rebase.sh | 15 + kubespray/tests/scripts/terraform_install.sh | 6 + kubespray/tests/scripts/testcases_cleanup.sh | 9 + kubespray/tests/scripts/testcases_prepare.sh | 7 + kubespray/tests/scripts/testcases_run.sh | 174 + kubespray/tests/scripts/vagrant-validate.sh | 6 + kubespray/tests/scripts/vagrant_clean.sh | 17 + kubespray/tests/shebang-unit | 1146 +++++ kubespray/tests/support/aws.groovy | 94 + kubespray/tests/templates/fake_hosts.yml.j2 | 3 + kubespray/tests/templates/inventory-aws.j2 | 29 + kubespray/tests/templates/inventory-do.j2 | 47 + kubespray/tests/templates/inventory-gce.j2 | 73 + .../tests/testcases/010_check-apiserver.yml | 24 + .../tests/testcases/015_check-nodes-ready.yml | 36 + .../testcases/020_check-pods-running.yml | 50 + .../tests/testcases/030_check-network.yml | 205 + .../tests/testcases/040_check-network-adv.yml | 241 + .../testcases/100_check-k8s-conformance.yml | 38 + .../roles/cluster-dump/tasks/main.yml | 19 + kubespray/upgrade-cluster.yml | 3 + 1430 files changed, 87701 insertions(+) create mode 100644 kubespray/Dockerfile create mode 100644 kubespray/Makefile create mode 100644 kubespray/README.md create mode 100644 kubespray/SECURITY_CONTACTS create mode 100644 kubespray/Vagrantfile create mode 100644 kubespray/ansible.cfg create mode 100644 kubespray/cluster.yml create mode 100644 kubespray/contrib/aws_iam/kubernetes-master-policy.json create mode 100644 kubespray/contrib/aws_iam/kubernetes-master-role.json create mode 100644 kubespray/contrib/aws_iam/kubernetes-minion-policy.json create mode 100644 kubespray/contrib/aws_iam/kubernetes-minion-role.json create mode 100755 kubespray/contrib/aws_inventory/kubespray-aws-inventory.py create mode 100644 kubespray/contrib/aws_inventory/requirements.txt create mode 100644 kubespray/contrib/azurerm/.gitignore create mode 100644 kubespray/contrib/azurerm/README.md create mode 100755 kubespray/contrib/azurerm/apply-rg.sh create mode 100755 kubespray/contrib/azurerm/clear-rg.sh create mode 100755 kubespray/contrib/azurerm/generate-inventory.sh create mode 100644 kubespray/contrib/azurerm/generate-inventory.yml create mode 100644 kubespray/contrib/azurerm/generate-inventory_2.yml create mode 100644 kubespray/contrib/azurerm/generate-templates.yml create mode 100644 kubespray/contrib/azurerm/group_vars/all create mode 100644 kubespray/contrib/azurerm/roles/generate-inventory/tasks/main.yml create mode 100644 kubespray/contrib/azurerm/roles/generate-inventory/templates/inventory.j2 create mode 100644 kubespray/contrib/azurerm/roles/generate-inventory_2/tasks/main.yml create mode 100644 kubespray/contrib/azurerm/roles/generate-inventory_2/templates/inventory.j2 create mode 100644 kubespray/contrib/azurerm/roles/generate-inventory_2/templates/loadbalancer_vars.j2 create mode 100644 kubespray/contrib/azurerm/roles/generate-templates/defaults/main.yml create mode 100644 kubespray/contrib/azurerm/roles/generate-templates/tasks/main.yml create mode 100644 kubespray/contrib/azurerm/roles/generate-templates/templates/availability-sets.json create mode 100644 kubespray/contrib/azurerm/roles/generate-templates/templates/bastion.json create mode 100644 kubespray/contrib/azurerm/roles/generate-templates/templates/clear-rg.json create mode 100644 kubespray/contrib/azurerm/roles/generate-templates/templates/masters.json create mode 100644 kubespray/contrib/azurerm/roles/generate-templates/templates/minions.json create mode 100644 kubespray/contrib/azurerm/roles/generate-templates/templates/network.json create mode 100644 kubespray/contrib/azurerm/roles/generate-templates/templates/storage.json create mode 100644 kubespray/contrib/dind/README.md create mode 100644 kubespray/contrib/dind/dind-cluster.yaml create mode 100644 kubespray/contrib/dind/group_vars/all/all.yaml create mode 100644 kubespray/contrib/dind/group_vars/all/distro.yaml create mode 100644 kubespray/contrib/dind/hosts create mode 100644 kubespray/contrib/dind/kubespray-dind.yaml create mode 100644 kubespray/contrib/dind/requirements.txt create mode 100644 kubespray/contrib/dind/roles/dind-cluster/tasks/main.yaml create mode 100644 kubespray/contrib/dind/roles/dind-host/tasks/main.yaml create mode 100644 kubespray/contrib/dind/roles/dind-host/templates/inventory_builder.sh.j2 create mode 100755 kubespray/contrib/dind/run-test-distros.sh create mode 100644 kubespray/contrib/dind/test-most_distros-some_CNIs.env create mode 100644 kubespray/contrib/dind/test-some_distros-kube_router_combo.env create mode 100644 kubespray/contrib/dind/test-some_distros-most_CNIs.env create mode 100644 kubespray/contrib/inventory_builder/inventory.py create mode 100644 kubespray/contrib/inventory_builder/requirements.txt create mode 100644 kubespray/contrib/inventory_builder/setup.cfg create mode 100644 kubespray/contrib/inventory_builder/setup.py create mode 100644 kubespray/contrib/inventory_builder/test-requirements.txt create mode 100644 kubespray/contrib/inventory_builder/tests/test_inventory.py create mode 100644 kubespray/contrib/inventory_builder/tox.ini create mode 100644 kubespray/contrib/kvm-setup/README.md create mode 100644 kubespray/contrib/kvm-setup/group_vars/all create mode 100644 kubespray/contrib/kvm-setup/kvm-setup.yml create mode 100644 kubespray/contrib/kvm-setup/roles/kvm-setup/tasks/main.yml create mode 100644 kubespray/contrib/kvm-setup/roles/kvm-setup/tasks/sysctl.yml create mode 100644 kubespray/contrib/kvm-setup/roles/kvm-setup/tasks/user.yml create mode 100644 kubespray/contrib/misc/clusteradmin-rbac.yml create mode 100644 kubespray/contrib/mitogen/mitogen.yml create mode 100644 kubespray/contrib/network-storage/glusterfs/README.md create mode 100644 kubespray/contrib/network-storage/glusterfs/glusterfs.yml create mode 120000 kubespray/contrib/network-storage/glusterfs/group_vars create mode 100644 kubespray/contrib/network-storage/glusterfs/inventory.example create mode 120000 kubespray/contrib/network-storage/glusterfs/roles/bootstrap-os create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/README.md create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/defaults/main.yml create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/meta/main.yml create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/tasks/main.yml create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/tasks/setup-Debian.yml create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/tasks/setup-RedHat.yml create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/defaults/main.yml create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/meta/main.yml create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/tasks/main.yml create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/tasks/setup-Debian.yml create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/tasks/setup-RedHat.yml create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/templates/test-file.txt create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/vars/Debian.yml create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/vars/RedHat.yml create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/tasks/main.yaml create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/templates/glusterfs-kubernetes-endpoint-svc.json.j2 create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/templates/glusterfs-kubernetes-endpoint.json.j2 create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/templates/glusterfs-kubernetes-pv.yml.j2 create mode 100644 kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/meta/main.yaml create mode 100644 kubespray/contrib/network-storage/heketi/README.md create mode 100644 kubespray/contrib/network-storage/heketi/heketi-tear-down.yml create mode 100644 kubespray/contrib/network-storage/heketi/heketi.yml create mode 100644 kubespray/contrib/network-storage/heketi/inventory.yml.sample create mode 100644 kubespray/contrib/network-storage/heketi/requirements.txt create mode 100644 kubespray/contrib/network-storage/heketi/roles/prepare/tasks/main.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/defaults/main.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/handlers/main.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/deploy.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/storage.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/tear-down.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/topology.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/volumes.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/cleanup.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/glusterfs.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/glusterfs/label.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/heketi.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/main.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/secret.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/storage.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/storageclass.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/tasks/topology.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/templates/glusterfs-daemonset.json.j2 create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-bootstrap.json.j2 create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-deployment.json.j2 create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-service-account.json.j2 create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-storage.json.j2 create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi.json.j2 create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/templates/storageclass.yml.j2 create mode 100644 kubespray/contrib/network-storage/heketi/roles/provision/templates/topology.json.j2 create mode 100644 kubespray/contrib/network-storage/heketi/roles/tear-down-disks/defaults/main.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/tear-down-disks/tasks/main.yml create mode 100644 kubespray/contrib/network-storage/heketi/roles/tear-down/tasks/main.yml create mode 100644 kubespray/contrib/offline/README.md create mode 100644 kubespray/contrib/offline/docker-daemon.json create mode 100755 kubespray/contrib/offline/generate_list.sh create mode 100644 kubespray/contrib/offline/generate_list.yml create mode 100755 kubespray/contrib/offline/manage-offline-container-images.sh create mode 100755 kubespray/contrib/offline/manage-offline-files.sh create mode 100644 kubespray/contrib/offline/nginx.conf create mode 100644 kubespray/contrib/offline/registries.conf create mode 100644 kubespray/contrib/os-services/os-services.yml create mode 100644 kubespray/contrib/os-services/roles/prepare/defaults/main.yml create mode 100644 kubespray/contrib/os-services/roles/prepare/tasks/main.yml create mode 100644 kubespray/contrib/packaging/rpm/kubespray.spec create mode 100644 kubespray/contrib/terraform/OWNERS create mode 100644 kubespray/contrib/terraform/aws/.gitignore create mode 100644 kubespray/contrib/terraform/aws/README.md create mode 100644 kubespray/contrib/terraform/aws/create-infrastructure.tf create mode 100644 kubespray/contrib/terraform/aws/credentials.tfvars.example create mode 100644 kubespray/contrib/terraform/aws/docs/aws_kubespray.png create mode 100644 kubespray/contrib/terraform/aws/modules/iam/main.tf create mode 100644 kubespray/contrib/terraform/aws/modules/iam/outputs.tf create mode 100644 kubespray/contrib/terraform/aws/modules/iam/variables.tf create mode 100644 kubespray/contrib/terraform/aws/modules/nlb/main.tf create mode 100644 kubespray/contrib/terraform/aws/modules/nlb/outputs.tf create mode 100644 kubespray/contrib/terraform/aws/modules/nlb/variables.tf create mode 100644 kubespray/contrib/terraform/aws/modules/vpc/main.tf create mode 100644 kubespray/contrib/terraform/aws/modules/vpc/outputs.tf create mode 100644 kubespray/contrib/terraform/aws/modules/vpc/variables.tf create mode 100644 kubespray/contrib/terraform/aws/output.tf create mode 100644 kubespray/contrib/terraform/aws/sample-inventory/cluster.tfvars create mode 120000 kubespray/contrib/terraform/aws/sample-inventory/group_vars create mode 100644 kubespray/contrib/terraform/aws/templates/inventory.tpl create mode 100644 kubespray/contrib/terraform/aws/terraform.tfvars create mode 100644 kubespray/contrib/terraform/aws/terraform.tfvars.example create mode 100644 kubespray/contrib/terraform/aws/variables.tf create mode 100644 kubespray/contrib/terraform/equinix/README.md create mode 120000 kubespray/contrib/terraform/equinix/hosts create mode 100644 kubespray/contrib/terraform/equinix/kubespray.tf create mode 100644 kubespray/contrib/terraform/equinix/output.tf create mode 100644 kubespray/contrib/terraform/equinix/provider.tf create mode 100644 kubespray/contrib/terraform/equinix/sample-inventory/cluster.tfvars create mode 120000 kubespray/contrib/terraform/equinix/sample-inventory/group_vars create mode 100644 kubespray/contrib/terraform/equinix/variables.tf create mode 100644 kubespray/contrib/terraform/exoscale/README.md create mode 100644 kubespray/contrib/terraform/exoscale/default.tfvars create mode 100644 kubespray/contrib/terraform/exoscale/main.tf create mode 100644 kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/main.tf create mode 100644 kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/output.tf create mode 100644 kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/templates/cloud-init.tmpl create mode 100644 kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/variables.tf create mode 100644 kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/versions.tf create mode 100644 kubespray/contrib/terraform/exoscale/output.tf create mode 100644 kubespray/contrib/terraform/exoscale/sample-inventory/cluster.tfvars create mode 120000 kubespray/contrib/terraform/exoscale/sample-inventory/group_vars create mode 100644 kubespray/contrib/terraform/exoscale/templates/inventory.tpl create mode 100644 kubespray/contrib/terraform/exoscale/variables.tf create mode 100644 kubespray/contrib/terraform/exoscale/versions.tf create mode 100644 kubespray/contrib/terraform/gcp/README.md create mode 100755 kubespray/contrib/terraform/gcp/generate-inventory.sh create mode 100644 kubespray/contrib/terraform/gcp/main.tf create mode 100644 kubespray/contrib/terraform/gcp/modules/kubernetes-cluster/main.tf create mode 100644 kubespray/contrib/terraform/gcp/modules/kubernetes-cluster/output.tf create mode 100644 kubespray/contrib/terraform/gcp/modules/kubernetes-cluster/variables.tf create mode 100644 kubespray/contrib/terraform/gcp/output.tf create mode 100644 kubespray/contrib/terraform/gcp/tfvars.json create mode 100644 kubespray/contrib/terraform/gcp/variables.tf create mode 120000 kubespray/contrib/terraform/group_vars create mode 100644 kubespray/contrib/terraform/hetzner/README.md create mode 100644 kubespray/contrib/terraform/hetzner/default.tfvars create mode 100644 kubespray/contrib/terraform/hetzner/main.tf create mode 100644 kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/main.tf create mode 100644 kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/outputs.tf create mode 100644 kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/templates/machine.yaml.tmpl create mode 100644 kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/variables.tf create mode 100644 kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/versions.tf create mode 100644 kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/main.tf create mode 100644 kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/output.tf create mode 100644 kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/templates/cloud-init.tmpl create mode 100644 kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/variables.tf create mode 100644 kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/versions.tf create mode 100644 kubespray/contrib/terraform/hetzner/output.tf create mode 100644 kubespray/contrib/terraform/hetzner/sample-inventory/cluster.tfvars create mode 120000 kubespray/contrib/terraform/hetzner/sample-inventory/group_vars create mode 100644 kubespray/contrib/terraform/hetzner/templates/inventory.tpl create mode 100644 kubespray/contrib/terraform/hetzner/variables.tf create mode 100644 kubespray/contrib/terraform/hetzner/versions.tf create mode 100644 kubespray/contrib/terraform/nifcloud/.gitignore create mode 100644 kubespray/contrib/terraform/nifcloud/README.md create mode 100755 kubespray/contrib/terraform/nifcloud/generate-inventory.sh create mode 100644 kubespray/contrib/terraform/nifcloud/main.tf create mode 100644 kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/main.tf create mode 100644 kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/outputs.tf create mode 100644 kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/templates/userdata.tftpl create mode 100644 kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/terraform.tf create mode 100644 kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/variables.tf create mode 100644 kubespray/contrib/terraform/nifcloud/output.tf create mode 100644 kubespray/contrib/terraform/nifcloud/sample-inventory/cluster.tfvars create mode 120000 kubespray/contrib/terraform/nifcloud/sample-inventory/group_vars create mode 100644 kubespray/contrib/terraform/nifcloud/terraform.tf create mode 100644 kubespray/contrib/terraform/nifcloud/variables.tf create mode 100644 kubespray/contrib/terraform/openstack/.gitignore create mode 100644 kubespray/contrib/terraform/openstack/README.md create mode 120000 kubespray/contrib/terraform/openstack/hosts create mode 100644 kubespray/contrib/terraform/openstack/kubespray.tf create mode 100644 kubespray/contrib/terraform/openstack/modules/compute/ansible_bastion_template.txt create mode 100644 kubespray/contrib/terraform/openstack/modules/compute/main.tf create mode 100644 kubespray/contrib/terraform/openstack/modules/compute/templates/cloudinit.yaml.tmpl create mode 100644 kubespray/contrib/terraform/openstack/modules/compute/variables.tf create mode 100644 kubespray/contrib/terraform/openstack/modules/compute/versions.tf create mode 100644 kubespray/contrib/terraform/openstack/modules/ips/main.tf create mode 100644 kubespray/contrib/terraform/openstack/modules/ips/outputs.tf create mode 100644 kubespray/contrib/terraform/openstack/modules/ips/variables.tf create mode 100644 kubespray/contrib/terraform/openstack/modules/ips/versions.tf create mode 100644 kubespray/contrib/terraform/openstack/modules/network/main.tf create mode 100644 kubespray/contrib/terraform/openstack/modules/network/outputs.tf create mode 100644 kubespray/contrib/terraform/openstack/modules/network/variables.tf create mode 100644 kubespray/contrib/terraform/openstack/modules/network/versions.tf create mode 100644 kubespray/contrib/terraform/openstack/sample-inventory/cluster.tfvars create mode 120000 kubespray/contrib/terraform/openstack/sample-inventory/group_vars create mode 100644 kubespray/contrib/terraform/openstack/variables.tf create mode 100644 kubespray/contrib/terraform/openstack/versions.tf create mode 100755 kubespray/contrib/terraform/terraform.py create mode 100644 kubespray/contrib/terraform/upcloud/README.md create mode 100644 kubespray/contrib/terraform/upcloud/cluster-settings.tfvars create mode 100644 kubespray/contrib/terraform/upcloud/main.tf create mode 100644 kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/main.tf create mode 100644 kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/output.tf create mode 100644 kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/variables.tf create mode 100644 kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/versions.tf create mode 100644 kubespray/contrib/terraform/upcloud/output.tf create mode 100644 kubespray/contrib/terraform/upcloud/sample-inventory/cluster.tfvars create mode 120000 kubespray/contrib/terraform/upcloud/sample-inventory/group_vars create mode 100644 kubespray/contrib/terraform/upcloud/templates/inventory.tpl create mode 100644 kubespray/contrib/terraform/upcloud/variables.tf create mode 100644 kubespray/contrib/terraform/upcloud/versions.tf create mode 100644 kubespray/contrib/terraform/vsphere/README.md create mode 100644 kubespray/contrib/terraform/vsphere/default.tfvars create mode 100644 kubespray/contrib/terraform/vsphere/main.tf create mode 100644 kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/main.tf create mode 100644 kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/output.tf create mode 100644 kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/templates/cloud-init.tpl create mode 100644 kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/templates/metadata.tpl create mode 100644 kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/templates/vapp-cloud-init.tpl create mode 100644 kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/variables.tf create mode 100644 kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/versions.tf create mode 100644 kubespray/contrib/terraform/vsphere/output.tf create mode 100644 kubespray/contrib/terraform/vsphere/sample-inventory/cluster.tfvars create mode 120000 kubespray/contrib/terraform/vsphere/sample-inventory/group_vars create mode 100644 kubespray/contrib/terraform/vsphere/templates/inventory.tpl create mode 100644 kubespray/contrib/terraform/vsphere/variables.tf create mode 100644 kubespray/contrib/terraform/vsphere/versions.tf create mode 100644 kubespray/docs/_sidebar.md create mode 100644 kubespray/docs/amazonlinux.md create mode 100644 kubespray/docs/ansible.md create mode 100644 kubespray/docs/ansible_collection.md create mode 100644 kubespray/docs/arch.md create mode 100644 kubespray/docs/aws-ebs-csi.md create mode 100644 kubespray/docs/aws.md create mode 100644 kubespray/docs/azure-csi.md create mode 100644 kubespray/docs/azure.md create mode 100644 kubespray/docs/bootstrap-os.md create mode 100644 kubespray/docs/calico.md create mode 100644 kubespray/docs/calico_peer_example/new-york.yml create mode 100644 kubespray/docs/calico_peer_example/paris.yml create mode 100644 kubespray/docs/centos.md create mode 100644 kubespray/docs/cert_manager.md create mode 100644 kubespray/docs/cgroups.md create mode 100644 kubespray/docs/ci-setup.md create mode 100644 kubespray/docs/ci.md create mode 100644 kubespray/docs/cilium.md create mode 100644 kubespray/docs/cinder-csi.md create mode 100644 kubespray/docs/cloud.md create mode 100644 kubespray/docs/cni.md create mode 100644 kubespray/docs/comparisons.md create mode 100644 kubespray/docs/containerd.md create mode 100644 kubespray/docs/cri-o.md create mode 100644 kubespray/docs/debian.md create mode 100644 kubespray/docs/dns-stack.md create mode 100644 kubespray/docs/docker.md create mode 100644 kubespray/docs/downloads.md create mode 100644 kubespray/docs/encrypting-secret-data-at-rest.md create mode 100644 kubespray/docs/equinix-metal.md create mode 100644 kubespray/docs/etcd.md create mode 100644 kubespray/docs/fcos.md create mode 100644 kubespray/docs/figures/kubespray-calico-rr.png create mode 100644 kubespray/docs/figures/loadbalancer_localhost.png create mode 100644 kubespray/docs/flannel.md create mode 100644 kubespray/docs/flatcar.md create mode 100644 kubespray/docs/gcp-lb.md create mode 100644 kubespray/docs/gcp-pd-csi.md create mode 100644 kubespray/docs/getting-started.md create mode 100644 kubespray/docs/gvisor.md create mode 100644 kubespray/docs/ha-mode.md create mode 100644 kubespray/docs/hardening.md create mode 100644 kubespray/docs/img/kubelet-hardening.png create mode 100644 kubespray/docs/img/kubernetes-logo.png create mode 100644 kubespray/docs/ingress_controller/alb_ingress_controller.md create mode 100644 kubespray/docs/ingress_controller/ingress_nginx.md create mode 100644 kubespray/docs/integration.md create mode 100644 kubespray/docs/kata-containers.md create mode 100644 kubespray/docs/kube-ovn.md create mode 100644 kubespray/docs/kube-router.md create mode 100644 kubespray/docs/kube-vip.md create mode 100644 kubespray/docs/kubernetes-apps/cephfs_provisioner.md create mode 100644 kubespray/docs/kubernetes-apps/local_volume_provisioner.md create mode 100644 kubespray/docs/kubernetes-apps/rbd_provisioner.md create mode 100644 kubespray/docs/kubernetes-apps/registry.md create mode 100644 kubespray/docs/kubernetes-reliability.md create mode 100644 kubespray/docs/kylinlinux.md create mode 100644 kubespray/docs/large-deployments.md create mode 100644 kubespray/docs/macvlan.md create mode 100644 kubespray/docs/metallb.md create mode 100644 kubespray/docs/mirror.md create mode 100644 kubespray/docs/mitogen.md create mode 100644 kubespray/docs/multus.md create mode 100644 kubespray/docs/netcheck.md create mode 100644 kubespray/docs/nodes.md create mode 100644 kubespray/docs/ntp.md create mode 100644 kubespray/docs/offline-environment.md create mode 100644 kubespray/docs/openeuler.md create mode 100644 kubespray/docs/openstack.md create mode 100644 kubespray/docs/opensuse.md create mode 100644 kubespray/docs/port-requirements.md create mode 100644 kubespray/docs/proxy.md create mode 100644 kubespray/docs/recover-control-plane.md create mode 100644 kubespray/docs/rhel.md create mode 100644 kubespray/docs/roadmap.md create mode 100644 kubespray/docs/setting-up-your-first-cluster.md create mode 100644 kubespray/docs/test_cases.md create mode 100644 kubespray/docs/uoslinux.md create mode 100644 kubespray/docs/upgrades.md create mode 100644 kubespray/docs/upgrades/migrate_docker2containerd.md create mode 100644 kubespray/docs/vagrant.md create mode 100644 kubespray/docs/vars.md create mode 100644 kubespray/docs/vsphere-csi.md create mode 100644 kubespray/docs/vsphere.md create mode 100644 kubespray/docs/weave.md create mode 100644 kubespray/extra_playbooks/files/get_cinder_pvs.sh create mode 120000 kubespray/extra_playbooks/inventory create mode 100644 kubespray/extra_playbooks/migrate_openstack_provider.yml create mode 120000 kubespray/extra_playbooks/roles create mode 100644 kubespray/extra_playbooks/upgrade-only-k8s.yml create mode 100644 kubespray/extra_playbooks/wait-for-cloud-init.yml create mode 120000 kubespray/inventory/local/group_vars create mode 100644 kubespray/inventory/local/hosts.ini create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/all.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/aws.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/azure.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/containerd.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/coreos.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/cri-o.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/docker.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/etcd.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/gcp.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/hcloud.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/huaweicloud.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/oci.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/offline.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/openstack.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/upcloud.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/all/vsphere.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/etcd.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/addons.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-cluster.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-calico.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-cilium.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-flannel.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-kube-ovn.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-kube-router.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-macvlan.yml create mode 100644 kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-weave.yml create mode 100644 kubespray/inventory/moaroom-cluster/inventory.ini create mode 100644 kubespray/inventory/moaroom-cluster/patches/kube-controller-manager+merge.yaml create mode 100644 kubespray/inventory/moaroom-cluster/patches/kube-scheduler+merge.yaml create mode 100644 kubespray/inventory/sample/group_vars/all/all.yml create mode 100644 kubespray/inventory/sample/group_vars/all/aws.yml create mode 100644 kubespray/inventory/sample/group_vars/all/azure.yml create mode 100644 kubespray/inventory/sample/group_vars/all/containerd.yml create mode 100644 kubespray/inventory/sample/group_vars/all/coreos.yml create mode 100644 kubespray/inventory/sample/group_vars/all/cri-o.yml create mode 100644 kubespray/inventory/sample/group_vars/all/docker.yml create mode 100644 kubespray/inventory/sample/group_vars/all/etcd.yml create mode 100644 kubespray/inventory/sample/group_vars/all/gcp.yml create mode 100644 kubespray/inventory/sample/group_vars/all/hcloud.yml create mode 100644 kubespray/inventory/sample/group_vars/all/huaweicloud.yml create mode 100644 kubespray/inventory/sample/group_vars/all/oci.yml create mode 100644 kubespray/inventory/sample/group_vars/all/offline.yml create mode 100644 kubespray/inventory/sample/group_vars/all/openstack.yml create mode 100644 kubespray/inventory/sample/group_vars/all/upcloud.yml create mode 100644 kubespray/inventory/sample/group_vars/all/vsphere.yml create mode 100644 kubespray/inventory/sample/group_vars/etcd.yml create mode 100644 kubespray/inventory/sample/group_vars/k8s_cluster/addons.yml create mode 100644 kubespray/inventory/sample/group_vars/k8s_cluster/k8s-cluster.yml create mode 100644 kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-calico.yml create mode 100644 kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-cilium.yml create mode 100644 kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-flannel.yml create mode 100644 kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-kube-ovn.yml create mode 100644 kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-kube-router.yml create mode 100644 kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-macvlan.yml create mode 100644 kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-weave.yml create mode 100644 kubespray/inventory/sample/inventory.ini create mode 100644 kubespray/inventory/sample/patches/kube-controller-manager+merge.yaml create mode 100644 kubespray/inventory/sample/patches/kube-scheduler+merge.yaml create mode 120000 kubespray/library/kube.py create mode 100644 kubespray/logo/LICENSE create mode 100644 kubespray/logo/OWNERS create mode 100644 kubespray/logo/logo-clear.png create mode 100644 kubespray/logo/logo-clear.svg create mode 100644 kubespray/logo/logo-dark.png create mode 100644 kubespray/logo/logo-dark.svg create mode 100644 kubespray/logo/logo-text-clear.png create mode 100644 kubespray/logo/logo-text-clear.svg create mode 100644 kubespray/logo/logo-text-dark.png create mode 100644 kubespray/logo/logo-text-dark.svg create mode 100644 kubespray/logo/logo-text-mixed.png create mode 100644 kubespray/logo/logo-text-mixed.svg create mode 100644 kubespray/logo/logos.pdf create mode 100644 kubespray/logo/usage_guidelines.md create mode 100644 kubespray/meta/runtime.yml create mode 100644 kubespray/pipeline.Dockerfile create mode 100644 kubespray/playbooks/ansible_version.yml create mode 100644 kubespray/playbooks/cluster.yml create mode 100644 kubespray/playbooks/facts.yml create mode 100644 kubespray/playbooks/legacy_groups.yml create mode 100644 kubespray/playbooks/recover_control_plane.yml create mode 100644 kubespray/playbooks/remove_node.yml create mode 100644 kubespray/playbooks/reset.yml create mode 100644 kubespray/playbooks/scale.yml create mode 100644 kubespray/playbooks/upgrade_cluster.yml create mode 100644 kubespray/plugins/modules/kube.py create mode 100644 kubespray/recover-control-plane.yml create mode 100644 kubespray/remove-node.yml create mode 100644 kubespray/requirements.txt create mode 100644 kubespray/reset.yml create mode 100644 kubespray/roles/adduser/defaults/main.yml create mode 100644 kubespray/roles/adduser/molecule/default/converge.yml create mode 100644 kubespray/roles/adduser/molecule/default/molecule.yml create mode 100644 kubespray/roles/adduser/molecule/default/tests/test_default.py create mode 100644 kubespray/roles/adduser/tasks/main.yml create mode 100644 kubespray/roles/adduser/vars/coreos.yml create mode 100644 kubespray/roles/adduser/vars/debian.yml create mode 100644 kubespray/roles/adduser/vars/redhat.yml create mode 100644 kubespray/roles/bastion-ssh-config/defaults/main.yml create mode 100644 kubespray/roles/bastion-ssh-config/molecule/default/converge.yml create mode 100644 kubespray/roles/bastion-ssh-config/molecule/default/molecule.yml create mode 100644 kubespray/roles/bastion-ssh-config/molecule/default/tests/test_default.py create mode 100644 kubespray/roles/bastion-ssh-config/tasks/main.yml create mode 100644 kubespray/roles/bastion-ssh-config/templates/ssh-bastion.conf.j2 create mode 100644 kubespray/roles/bootstrap-os/defaults/main.yml create mode 100755 kubespray/roles/bootstrap-os/files/bootstrap.sh create mode 100644 kubespray/roles/bootstrap-os/handlers/main.yml create mode 100644 kubespray/roles/bootstrap-os/molecule/default/converge.yml create mode 100644 kubespray/roles/bootstrap-os/molecule/default/molecule.yml create mode 100644 kubespray/roles/bootstrap-os/molecule/default/tests/test_default.py create mode 100644 kubespray/roles/bootstrap-os/tasks/bootstrap-amazon.yml create mode 100644 kubespray/roles/bootstrap-os/tasks/bootstrap-centos.yml create mode 100644 kubespray/roles/bootstrap-os/tasks/bootstrap-clearlinux.yml create mode 100644 kubespray/roles/bootstrap-os/tasks/bootstrap-coreos.yml create mode 100644 kubespray/roles/bootstrap-os/tasks/bootstrap-debian.yml create mode 100644 kubespray/roles/bootstrap-os/tasks/bootstrap-fedora-coreos.yml create mode 100644 kubespray/roles/bootstrap-os/tasks/bootstrap-fedora.yml create mode 100644 kubespray/roles/bootstrap-os/tasks/bootstrap-flatcar.yml create mode 100644 kubespray/roles/bootstrap-os/tasks/bootstrap-opensuse.yml create mode 100644 kubespray/roles/bootstrap-os/tasks/bootstrap-redhat.yml create mode 100644 kubespray/roles/bootstrap-os/tasks/main.yml create mode 100644 kubespray/roles/container-engine/containerd-common/defaults/main.yml create mode 100644 kubespray/roles/container-engine/containerd-common/meta/main.yml create mode 100644 kubespray/roles/container-engine/containerd-common/tasks/main.yml create mode 100644 kubespray/roles/container-engine/containerd-common/vars/amazon.yml create mode 100644 kubespray/roles/container-engine/containerd-common/vars/suse.yml create mode 100644 kubespray/roles/container-engine/containerd/defaults/main.yml create mode 100644 kubespray/roles/container-engine/containerd/handlers/main.yml create mode 100644 kubespray/roles/container-engine/containerd/handlers/reset.yml create mode 100644 kubespray/roles/container-engine/containerd/meta/main.yml create mode 100644 kubespray/roles/container-engine/containerd/molecule/default/converge.yml create mode 100644 kubespray/roles/container-engine/containerd/molecule/default/molecule.yml create mode 100644 kubespray/roles/container-engine/containerd/molecule/default/prepare.yml create mode 100644 kubespray/roles/container-engine/containerd/molecule/default/tests/test_default.py create mode 100644 kubespray/roles/container-engine/containerd/tasks/main.yml create mode 100644 kubespray/roles/container-engine/containerd/tasks/reset.yml create mode 100644 kubespray/roles/container-engine/containerd/templates/config.toml.j2 create mode 100644 kubespray/roles/container-engine/containerd/templates/containerd.service.j2 create mode 100644 kubespray/roles/container-engine/containerd/templates/hosts.toml.j2 create mode 100644 kubespray/roles/container-engine/containerd/templates/http-proxy.conf.j2 create mode 100644 kubespray/roles/container-engine/containerd/vars/debian.yml create mode 100644 kubespray/roles/container-engine/containerd/vars/ubuntu.yml create mode 100644 kubespray/roles/container-engine/cri-dockerd/handlers/main.yml create mode 100644 kubespray/roles/container-engine/cri-dockerd/meta/main.yml create mode 100644 kubespray/roles/container-engine/cri-dockerd/molecule/default/converge.yml create mode 100644 kubespray/roles/container-engine/cri-dockerd/molecule/default/files/10-mynet.conf create mode 100644 kubespray/roles/container-engine/cri-dockerd/molecule/default/files/container.json create mode 100644 kubespray/roles/container-engine/cri-dockerd/molecule/default/files/sandbox.json create mode 100644 kubespray/roles/container-engine/cri-dockerd/molecule/default/molecule.yml create mode 100644 kubespray/roles/container-engine/cri-dockerd/molecule/default/prepare.yml create mode 100644 kubespray/roles/container-engine/cri-dockerd/molecule/default/tests/test_default.py create mode 100644 kubespray/roles/container-engine/cri-dockerd/tasks/main.yml create mode 100644 kubespray/roles/container-engine/cri-dockerd/templates/cri-dockerd.service.j2 create mode 100644 kubespray/roles/container-engine/cri-dockerd/templates/cri-dockerd.socket.j2 create mode 100644 kubespray/roles/container-engine/cri-o/defaults/main.yml create mode 100644 kubespray/roles/container-engine/cri-o/files/mounts.conf create mode 100644 kubespray/roles/container-engine/cri-o/handlers/main.yml create mode 100644 kubespray/roles/container-engine/cri-o/meta/main.yml create mode 100644 kubespray/roles/container-engine/cri-o/molecule/default/converge.yml create mode 100644 kubespray/roles/container-engine/cri-o/molecule/default/files/10-mynet.conf create mode 100644 kubespray/roles/container-engine/cri-o/molecule/default/files/container.json create mode 100644 kubespray/roles/container-engine/cri-o/molecule/default/files/sandbox.json create mode 100644 kubespray/roles/container-engine/cri-o/molecule/default/molecule.yml create mode 100644 kubespray/roles/container-engine/cri-o/molecule/default/prepare.yml create mode 100644 kubespray/roles/container-engine/cri-o/molecule/default/tests/test_default.py create mode 100644 kubespray/roles/container-engine/cri-o/tasks/cleanup.yaml create mode 100644 kubespray/roles/container-engine/cri-o/tasks/main.yaml create mode 100644 kubespray/roles/container-engine/cri-o/tasks/reset.yml create mode 100644 kubespray/roles/container-engine/cri-o/tasks/setup-amazon.yaml create mode 100644 kubespray/roles/container-engine/cri-o/templates/config.json.j2 create mode 100644 kubespray/roles/container-engine/cri-o/templates/crio.conf.j2 create mode 100644 kubespray/roles/container-engine/cri-o/templates/http-proxy.conf.j2 create mode 100644 kubespray/roles/container-engine/cri-o/templates/registry.conf.j2 create mode 100644 kubespray/roles/container-engine/cri-o/templates/unqualified.conf.j2 create mode 100644 kubespray/roles/container-engine/crictl/handlers/main.yml create mode 100644 kubespray/roles/container-engine/crictl/tasks/crictl.yml create mode 100644 kubespray/roles/container-engine/crictl/tasks/main.yml create mode 100644 kubespray/roles/container-engine/crictl/templates/crictl.yaml.j2 create mode 100644 kubespray/roles/container-engine/crun/tasks/main.yml create mode 100644 kubespray/roles/container-engine/docker-storage/defaults/main.yml create mode 100644 kubespray/roles/container-engine/docker-storage/files/install_container_storage_setup.sh create mode 100644 kubespray/roles/container-engine/docker-storage/tasks/main.yml create mode 100644 kubespray/roles/container-engine/docker-storage/templates/docker-storage-setup.j2 create mode 100644 kubespray/roles/container-engine/docker/defaults/main.yml create mode 100644 kubespray/roles/container-engine/docker/files/cleanup-docker-orphans.sh create mode 100644 kubespray/roles/container-engine/docker/handlers/main.yml create mode 100644 kubespray/roles/container-engine/docker/meta/main.yml create mode 100644 kubespray/roles/container-engine/docker/tasks/docker_plugin.yml create mode 100644 kubespray/roles/container-engine/docker/tasks/main.yml create mode 100644 kubespray/roles/container-engine/docker/tasks/pre-upgrade.yml create mode 100644 kubespray/roles/container-engine/docker/tasks/reset.yml create mode 100644 kubespray/roles/container-engine/docker/tasks/set_facts_dns.yml create mode 100644 kubespray/roles/container-engine/docker/tasks/systemd.yml create mode 100644 kubespray/roles/container-engine/docker/templates/docker-dns.conf.j2 create mode 100644 kubespray/roles/container-engine/docker/templates/docker-options.conf.j2 create mode 100644 kubespray/roles/container-engine/docker/templates/docker-orphan-cleanup.conf.j2 create mode 100644 kubespray/roles/container-engine/docker/templates/docker.service.j2 create mode 100644 kubespray/roles/container-engine/docker/templates/fedora_docker.repo.j2 create mode 100644 kubespray/roles/container-engine/docker/templates/http-proxy.conf.j2 create mode 100644 kubespray/roles/container-engine/docker/templates/rh_docker.repo.j2 create mode 100644 kubespray/roles/container-engine/docker/vars/amazon.yml create mode 100644 kubespray/roles/container-engine/docker/vars/clearlinux.yml create mode 100644 kubespray/roles/container-engine/docker/vars/debian-bookworm.yml create mode 100644 kubespray/roles/container-engine/docker/vars/debian.yml create mode 100644 kubespray/roles/container-engine/docker/vars/fedora.yml create mode 100644 kubespray/roles/container-engine/docker/vars/kylin.yml create mode 100644 kubespray/roles/container-engine/docker/vars/redhat-7.yml create mode 100644 kubespray/roles/container-engine/docker/vars/redhat.yml create mode 100644 kubespray/roles/container-engine/docker/vars/suse.yml create mode 100644 kubespray/roles/container-engine/docker/vars/ubuntu.yml create mode 100644 kubespray/roles/container-engine/docker/vars/uniontech.yml create mode 100644 kubespray/roles/container-engine/gvisor/molecule/default/converge.yml create mode 100644 kubespray/roles/container-engine/gvisor/molecule/default/files/10-mynet.conf create mode 100644 kubespray/roles/container-engine/gvisor/molecule/default/files/container.json create mode 100644 kubespray/roles/container-engine/gvisor/molecule/default/files/sandbox.json create mode 100644 kubespray/roles/container-engine/gvisor/molecule/default/molecule.yml create mode 100644 kubespray/roles/container-engine/gvisor/molecule/default/prepare.yml create mode 100644 kubespray/roles/container-engine/gvisor/molecule/default/tests/test_default.py create mode 100644 kubespray/roles/container-engine/gvisor/tasks/main.yml create mode 100644 kubespray/roles/container-engine/kata-containers/OWNERS create mode 100644 kubespray/roles/container-engine/kata-containers/defaults/main.yml create mode 100644 kubespray/roles/container-engine/kata-containers/molecule/default/converge.yml create mode 100644 kubespray/roles/container-engine/kata-containers/molecule/default/files/10-mynet.conf create mode 100644 kubespray/roles/container-engine/kata-containers/molecule/default/files/container.json create mode 100644 kubespray/roles/container-engine/kata-containers/molecule/default/files/sandbox.json create mode 100644 kubespray/roles/container-engine/kata-containers/molecule/default/molecule.yml create mode 100644 kubespray/roles/container-engine/kata-containers/molecule/default/prepare.yml create mode 100644 kubespray/roles/container-engine/kata-containers/molecule/default/tests/test_default.py create mode 100644 kubespray/roles/container-engine/kata-containers/tasks/main.yml create mode 100644 kubespray/roles/container-engine/kata-containers/templates/configuration-qemu.toml.j2 create mode 100644 kubespray/roles/container-engine/kata-containers/templates/containerd-shim-kata-v2.j2 create mode 100644 kubespray/roles/container-engine/meta/main.yml create mode 100644 kubespray/roles/container-engine/nerdctl/handlers/main.yml create mode 100644 kubespray/roles/container-engine/nerdctl/tasks/main.yml create mode 100644 kubespray/roles/container-engine/nerdctl/templates/nerdctl.toml.j2 create mode 100644 kubespray/roles/container-engine/runc/defaults/main.yml create mode 100644 kubespray/roles/container-engine/runc/tasks/main.yml create mode 100644 kubespray/roles/container-engine/skopeo/tasks/main.yml create mode 100644 kubespray/roles/container-engine/validate-container-engine/tasks/main.yml create mode 100644 kubespray/roles/container-engine/youki/defaults/main.yml create mode 100644 kubespray/roles/container-engine/youki/molecule/default/converge.yml create mode 100644 kubespray/roles/container-engine/youki/molecule/default/files/10-mynet.conf create mode 100644 kubespray/roles/container-engine/youki/molecule/default/files/container.json create mode 100644 kubespray/roles/container-engine/youki/molecule/default/files/sandbox.json create mode 100644 kubespray/roles/container-engine/youki/molecule/default/molecule.yml create mode 100644 kubespray/roles/container-engine/youki/molecule/default/prepare.yml create mode 100644 kubespray/roles/container-engine/youki/molecule/default/tests/test_default.py create mode 100644 kubespray/roles/container-engine/youki/tasks/main.yml create mode 100644 kubespray/roles/download/defaults/main/checksums.yml create mode 100644 kubespray/roles/download/defaults/main/main.yml create mode 100644 kubespray/roles/download/meta/main.yml create mode 100644 kubespray/roles/download/tasks/check_pull_required.yml create mode 100644 kubespray/roles/download/tasks/download_container.yml create mode 100644 kubespray/roles/download/tasks/download_file.yml create mode 100644 kubespray/roles/download/tasks/extract_file.yml create mode 100644 kubespray/roles/download/tasks/main.yml create mode 100644 kubespray/roles/download/tasks/prep_download.yml create mode 100644 kubespray/roles/download/tasks/prep_kubeadm_images.yml create mode 100644 kubespray/roles/download/tasks/set_container_facts.yml create mode 100644 kubespray/roles/download/templates/kubeadm-images.yaml.j2 create mode 100644 kubespray/roles/etcd/defaults/main.yml create mode 100644 kubespray/roles/etcd/handlers/backup.yml create mode 100644 kubespray/roles/etcd/handlers/backup_cleanup.yml create mode 100644 kubespray/roles/etcd/handlers/main.yml create mode 100644 kubespray/roles/etcd/meta/main.yml create mode 100644 kubespray/roles/etcd/tasks/check_certs.yml create mode 100644 kubespray/roles/etcd/tasks/configure.yml create mode 100644 kubespray/roles/etcd/tasks/gen_certs_script.yml create mode 100644 kubespray/roles/etcd/tasks/gen_nodes_certs_script.yml create mode 100644 kubespray/roles/etcd/tasks/install_docker.yml create mode 100644 kubespray/roles/etcd/tasks/install_host.yml create mode 100644 kubespray/roles/etcd/tasks/join_etcd-events_member.yml create mode 100644 kubespray/roles/etcd/tasks/join_etcd_member.yml create mode 100644 kubespray/roles/etcd/tasks/main.yml create mode 100644 kubespray/roles/etcd/tasks/refresh_config.yml create mode 100644 kubespray/roles/etcd/tasks/upd_ca_trust.yml create mode 100644 kubespray/roles/etcd/templates/etcd-docker.service.j2 create mode 100644 kubespray/roles/etcd/templates/etcd-events-docker.service.j2 create mode 100644 kubespray/roles/etcd/templates/etcd-events-host.service.j2 create mode 100644 kubespray/roles/etcd/templates/etcd-events.env.j2 create mode 100644 kubespray/roles/etcd/templates/etcd-events.j2 create mode 100644 kubespray/roles/etcd/templates/etcd-host.service.j2 create mode 100644 kubespray/roles/etcd/templates/etcd.env.j2 create mode 100644 kubespray/roles/etcd/templates/etcd.j2 create mode 100644 kubespray/roles/etcd/templates/make-ssl-etcd.sh.j2 create mode 100644 kubespray/roles/etcd/templates/openssl.conf.j2 create mode 100644 kubespray/roles/etcdctl_etcdutl/tasks/main.yml create mode 100644 kubespray/roles/etcdctl_etcdutl/templates/etcdctl.sh.j2 create mode 100644 kubespray/roles/helm-apps/README.md create mode 100644 kubespray/roles/helm-apps/meta/argument_specs.yml create mode 100644 kubespray/roles/helm-apps/meta/main.yml create mode 100644 kubespray/roles/helm-apps/tasks/main.yml create mode 100644 kubespray/roles/helm-apps/vars/main.yml create mode 100644 kubespray/roles/kubernetes-apps/ansible/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/ansible/tasks/cleanup_dns.yml create mode 100644 kubespray/roles/kubernetes-apps/ansible/tasks/coredns.yml create mode 100644 kubespray/roles/kubernetes-apps/ansible/tasks/dashboard.yml create mode 100644 kubespray/roles/kubernetes-apps/ansible/tasks/etcd_metrics.yml create mode 100644 kubespray/roles/kubernetes-apps/ansible/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/ansible/tasks/netchecker.yml create mode 100644 kubespray/roles/kubernetes-apps/ansible/tasks/nodelocaldns.yml create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/coredns-clusterrole.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/coredns-clusterrolebinding.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/coredns-config.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/coredns-deployment.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/coredns-sa.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/coredns-svc.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/dashboard.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler-clusterrole.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler-clusterrolebinding.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler-sa.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/etcd_metrics-endpoints.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/etcd_metrics-service.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-ds.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-clusterrole.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-clusterrolebinding.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-ds.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-psp.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-sa.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/netchecker-ns.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-clusterrole.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-clusterrolebinding.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-deployment.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-sa.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-svc.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-config.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-daemonset.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-sa.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-second-daemonset.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/argocd/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/argocd/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/argocd/templates/argocd-namespace.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/cloud_controller/oci/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/cloud_controller/oci/tasks/credentials-check.yml create mode 100644 kubespray/roles/kubernetes-apps/cloud_controller/oci/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/cloud_controller/oci/templates/controller-manager-config.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/cloud_controller/oci/templates/oci-cloud-provider.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/cluster_roles/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/cluster_roles/files/k8s-cluster-critical-pc.yml create mode 100644 kubespray/roles/kubernetes-apps/cluster_roles/files/oci-rbac.yml create mode 100644 kubespray/roles/kubernetes-apps/cluster_roles/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/cluster_roles/tasks/oci.yml create mode 100644 kubespray/roles/kubernetes-apps/cluster_roles/templates/namespace.j2 create mode 100644 kubespray/roles/kubernetes-apps/cluster_roles/templates/node-crb.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/cluster_roles/templates/vsphere-rbac.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/container_engine_accelerator/meta/main.yml create mode 100644 kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/templates/k8s-device-plugin-nvidia-daemonset.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/templates/nvidia-driver-install-daemonset.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/vars/centos-7.yml create mode 100644 kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/vars/ubuntu-16.yml create mode 100644 kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/vars/ubuntu-18.yml create mode 100644 kubespray/roles/kubernetes-apps/container_runtimes/crun/tasks/main.yaml create mode 100644 kubespray/roles/kubernetes-apps/container_runtimes/crun/templates/runtimeclass-crun.yml create mode 100644 kubespray/roles/kubernetes-apps/container_runtimes/gvisor/tasks/main.yaml create mode 100644 kubespray/roles/kubernetes-apps/container_runtimes/gvisor/templates/runtimeclass-gvisor.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/container_runtimes/kata_containers/defaults/main.yaml create mode 100644 kubespray/roles/kubernetes-apps/container_runtimes/kata_containers/tasks/main.yaml create mode 100644 kubespray/roles/kubernetes-apps/container_runtimes/kata_containers/templates/runtimeclass-kata-qemu.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/container_runtimes/meta/main.yml create mode 100644 kubespray/roles/kubernetes-apps/container_runtimes/youki/tasks/main.yaml create mode 100644 kubespray/roles/kubernetes-apps/container_runtimes/youki/templates/runtimeclass-youki.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/OWNERS create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-controllerservice-rbac.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-controllerservice.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-driver.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-nodeservice.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/azuredisk/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/azuredisk/tasks/azure-credential-check.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/azuredisk/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-controller-rbac.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-controller.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-driver.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-node-rbac.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-node.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-cloud-config-secret.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-cloud-config.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/cinder/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/cinder/tasks/cinder-credential-check.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/cinder/tasks/cinder-write-cacert.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/cinder/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-cloud-config-secret.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-cloud-config.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-controllerplugin-rbac.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-controllerplugin.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-driver.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-nodeplugin-rbac.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-nodeplugin.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-poddisruptionbudget.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/csi_crd/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/csi_crd/templates/volumesnapshotclasses.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/csi_crd/templates/volumesnapshotcontents.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/csi_crd/templates/volumesnapshots.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-controller.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-cred-secret.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-node.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-sc-regional.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-sc-zonal.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-setup.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/upcloud/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/upcloud/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-controller.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-cred-secret.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-driver.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-node.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-setup.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/vsphere/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/vsphere/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/vsphere/tasks/vsphere-credentials-check.yml create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-cloud-config.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-config.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-deployment.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-rbac.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-service.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-driver.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-namespace.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-node-rbac.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-node.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-controller-manager-ds-with-networks.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-controller-manager-ds.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-role-bindings.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-secret.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-service-account.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/tasks/huaweicloud-credential-check.yml create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-config-secret.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-config.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-controller-manager-ds.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-controller-manager-role-bindings.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-controller-manager-roles.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/meta/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/OWNERS create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/tasks/openstack-credential-check.yml create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-config-secret.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-config.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-controller-manager-ds.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-controller-manager-role-bindings.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-controller-manager-roles.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/tasks/vsphere-credentials-check.yml create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cloud-controller-manager-ds.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cloud-controller-manager-role-bindings.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cloud-controller-manager-roles.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cpi-cloud-config-secret.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cpi-cloud-config.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/00-namespace.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/clusterrole-cephfs-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/clusterrolebinding-cephfs-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/deploy-cephfs-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/psp-cephfs-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/role-cephfs-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/rolebinding-cephfs-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/sa-cephfs-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/sc-cephfs-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/secret-cephfs-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-clusterrolebinding.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-cm.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-cr.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-deployment.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-ns.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-sa.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-sc.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/tasks/basedirs.yml create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-clusterrole.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-clusterrolebinding.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-cm.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-ds.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-ns.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-sa.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-sc.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/meta/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/00-namespace.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/clusterrole-rbd-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/clusterrolebinding-rbd-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/deploy-rbd-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/psp-rbd-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/role-rbd-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/rolebinding-rbd-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/sa-rbd-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/sc-rbd-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/secret-rbd-provisioner.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/helm/.gitkeep create mode 100644 kubespray/roles/kubernetes-apps/helm/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/helm/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/helm/tasks/pyyaml-flatcar.yml create mode 100644 kubespray/roles/kubernetes-apps/helm/vars/amazon.yml create mode 100644 kubespray/roles/kubernetes-apps/helm/vars/centos-7.yml create mode 100644 kubespray/roles/kubernetes-apps/helm/vars/centos.yml create mode 100644 kubespray/roles/kubernetes-apps/helm/vars/debian.yml create mode 100644 kubespray/roles/kubernetes-apps/helm/vars/fedora.yml create mode 100644 kubespray/roles/kubernetes-apps/helm/vars/redhat-7.yml create mode 100644 kubespray/roles/kubernetes-apps/helm/vars/redhat.yml create mode 100644 kubespray/roles/kubernetes-apps/helm/vars/suse.yml create mode 100644 kubespray/roles/kubernetes-apps/helm/vars/ubuntu.yml create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/OWNERS create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-clusterrole.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-clusterrolebinding.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-deploy.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-ns.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-sa.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/templates/cert-manager.crds.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/templates/cert-manager.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/00-namespace.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/admission-webhook-configuration.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/admission-webhook-job.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrole-admission-webhook.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrole-ingress-nginx.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrolebinding-admission-webhook.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrolebinding-ingress-nginx.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/cm-ingress-nginx.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/cm-tcp-services.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/cm-udp-services.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/ds-ingress-nginx-controller.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/ingressclass-nginx.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/role-admission-webhook.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/role-ingress-nginx.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/rolebinding-admission-webhook.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/rolebinding-ingress-nginx.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/sa-admission-webhook.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/sa-ingress-nginx.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/ingress_controller/meta/main.yml create mode 100644 kubespray/roles/kubernetes-apps/krew/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/krew/tasks/krew.yml create mode 100644 kubespray/roles/kubernetes-apps/krew/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/krew/templates/krew.j2 create mode 100644 kubespray/roles/kubernetes-apps/krew/templates/krew.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/kubelet-csr-approver/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/kubelet-csr-approver/meta/main.yml create mode 100644 kubespray/roles/kubernetes-apps/meta/main.yml create mode 100644 kubespray/roles/kubernetes-apps/metallb/OWNERS create mode 100644 kubespray/roles/kubernetes-apps/metallb/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/metallb/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/metallb/templates/layer2.yaml.j2 create mode 100644 kubespray/roles/kubernetes-apps/metallb/templates/layer3.yaml.j2 create mode 100644 kubespray/roles/kubernetes-apps/metallb/templates/metallb.yaml.j2 create mode 100644 kubespray/roles/kubernetes-apps/metallb/templates/pools.yaml.j2 create mode 100644 kubespray/roles/kubernetes-apps/metrics_server/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/metrics_server/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/metrics_server/templates/auth-delegator.yaml.j2 create mode 100644 kubespray/roles/kubernetes-apps/metrics_server/templates/auth-reader.yaml.j2 create mode 100644 kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-apiservice.yaml.j2 create mode 100644 kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-server-deployment.yaml.j2 create mode 100644 kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-server-sa.yaml.j2 create mode 100644 kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-server-service.yaml.j2 create mode 100644 kubespray/roles/kubernetes-apps/metrics_server/templates/resource-reader-clusterrolebinding.yaml.j2 create mode 100644 kubespray/roles/kubernetes-apps/metrics_server/templates/resource-reader.yaml.j2 create mode 100644 kubespray/roles/kubernetes-apps/network_plugin/calico/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/network_plugin/flannel/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/network_plugin/kube-ovn/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/network_plugin/kube-router/OWNERS create mode 100644 kubespray/roles/kubernetes-apps/network_plugin/kube-router/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/network_plugin/meta/main.yml create mode 100644 kubespray/roles/kubernetes-apps/network_plugin/multus/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/network_plugin/weave/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/OWNERS create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/templates/aws-ebs-csi-storage-class.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/azuredisk-csi/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/azuredisk-csi/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/azuredisk-csi/templates/azure-csi-storage-class.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/cinder-csi/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/cinder-csi/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/cinder-csi/templates/cinder-csi-storage-class.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/gcp-pd-csi/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/gcp-pd-csi/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/gcp-pd-csi/templates/gcp-pd-csi-storage-class.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/meta/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/openstack/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/openstack/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/openstack/templates/openstack-storage-class.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/upcloud-csi/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/upcloud-csi/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/persistent_volumes/upcloud-csi/templates/upcloud-csi-storage-class.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/policy_controller/calico/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/policy_controller/calico/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-controllers.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-cr.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-crb.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-sa.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/policy_controller/meta/main.yml create mode 100644 kubespray/roles/kubernetes-apps/registry/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/registry/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/registry/templates/registry-cm.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/registry/templates/registry-cr.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/registry/templates/registry-crb.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/registry/templates/registry-ing.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/registry/templates/registry-ns.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/registry/templates/registry-psp.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/registry/templates/registry-pvc.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/registry/templates/registry-rs.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/registry/templates/registry-sa.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/registry/templates/registry-secrets.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/registry/templates/registry-svc.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/snapshots/cinder-csi/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/snapshots/cinder-csi/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/snapshots/cinder-csi/templates/cinder-csi-snapshot-class.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/snapshots/meta/main.yml create mode 100644 kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/defaults/main.yml create mode 100644 kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/tasks/main.yml create mode 100644 kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/templates/rbac-snapshot-controller.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/templates/snapshot-controller.yml.j2 create mode 100644 kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/templates/snapshot-ns.yml.j2 create mode 100644 kubespray/roles/kubernetes/client/defaults/main.yml create mode 100644 kubespray/roles/kubernetes/client/tasks/main.yml create mode 100644 kubespray/roles/kubernetes/control-plane/defaults/main/etcd.yml create mode 100644 kubespray/roles/kubernetes/control-plane/defaults/main/kube-proxy.yml create mode 100644 kubespray/roles/kubernetes/control-plane/defaults/main/kube-scheduler.yml create mode 100644 kubespray/roles/kubernetes/control-plane/defaults/main/main.yml create mode 100644 kubespray/roles/kubernetes/control-plane/handlers/main.yml create mode 100644 kubespray/roles/kubernetes/control-plane/meta/main.yml create mode 100644 kubespray/roles/kubernetes/control-plane/tasks/define-first-kube-control.yml create mode 100644 kubespray/roles/kubernetes/control-plane/tasks/encrypt-at-rest.yml create mode 100644 kubespray/roles/kubernetes/control-plane/tasks/kubeadm-backup.yml create mode 100644 kubespray/roles/kubernetes/control-plane/tasks/kubeadm-etcd.yml create mode 100644 kubespray/roles/kubernetes/control-plane/tasks/kubeadm-fix-apiserver.yml create mode 100644 kubespray/roles/kubernetes/control-plane/tasks/kubeadm-secondary.yml create mode 100644 kubespray/roles/kubernetes/control-plane/tasks/kubeadm-setup.yml create mode 100644 kubespray/roles/kubernetes/control-plane/tasks/kubeadm-upgrade.yml create mode 100644 kubespray/roles/kubernetes/control-plane/tasks/kubelet-fix-client-cert-rotation.yml create mode 100644 kubespray/roles/kubernetes/control-plane/tasks/main.yml create mode 100644 kubespray/roles/kubernetes/control-plane/tasks/pre-upgrade.yml create mode 100644 kubespray/roles/kubernetes/control-plane/tasks/psp-install.yml create mode 100644 kubespray/roles/kubernetes/control-plane/templates/admission-controls.yaml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/apiserver-audit-policy.yaml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/apiserver-audit-webhook-config.yaml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/eventratelimit.yaml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/k8s-certs-renew.service.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/k8s-certs-renew.sh.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/k8s-certs-renew.timer.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/kubeadm-config.v1beta3.yaml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/kubeadm-controlplane.v1beta3.yaml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/kubescheduler-config.yaml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/podsecurity.yaml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/psp-cr.yml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/psp-crb.yml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/psp.yml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/secrets_encryption.yaml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/webhook-authorization-config.yaml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/templates/webhook-token-auth-config.yaml.j2 create mode 100644 kubespray/roles/kubernetes/control-plane/vars/main.yaml create mode 100644 kubespray/roles/kubernetes/kubeadm/defaults/main.yml create mode 100644 kubespray/roles/kubernetes/kubeadm/handlers/main.yml create mode 100644 kubespray/roles/kubernetes/kubeadm/tasks/kubeadm_etcd_node.yml create mode 100644 kubespray/roles/kubernetes/kubeadm/tasks/main.yml create mode 100644 kubespray/roles/kubernetes/kubeadm/templates/kubeadm-client.conf.v1beta3.j2 create mode 100644 kubespray/roles/kubernetes/node-label/tasks/main.yml create mode 100644 kubespray/roles/kubernetes/node/defaults/main.yml create mode 100644 kubespray/roles/kubernetes/node/handlers/main.yml create mode 100644 kubespray/roles/kubernetes/node/tasks/cloud-credentials/azure-credential-check.yml create mode 100644 kubespray/roles/kubernetes/node/tasks/cloud-credentials/openstack-credential-check.yml create mode 100644 kubespray/roles/kubernetes/node/tasks/cloud-credentials/vsphere-credential-check.yml create mode 100644 kubespray/roles/kubernetes/node/tasks/facts.yml create mode 100644 kubespray/roles/kubernetes/node/tasks/install.yml create mode 100644 kubespray/roles/kubernetes/node/tasks/kubelet.yml create mode 100644 kubespray/roles/kubernetes/node/tasks/loadbalancer/haproxy.yml create mode 100644 kubespray/roles/kubernetes/node/tasks/loadbalancer/kube-vip.yml create mode 100644 kubespray/roles/kubernetes/node/tasks/loadbalancer/nginx-proxy.yml create mode 100644 kubespray/roles/kubernetes/node/tasks/main.yml create mode 100644 kubespray/roles/kubernetes/node/tasks/pre_upgrade.yml create mode 100644 kubespray/roles/kubernetes/node/templates/cloud-configs/aws-cloud-config.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/cloud-configs/azure-cloud-config.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/cloud-configs/gce-cloud-config.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/cloud-configs/openstack-cloud-config.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/cloud-configs/vsphere-cloud-config.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/http-proxy.conf.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/kubelet-config.v1beta1.yaml.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/kubelet.env.v1beta1.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/kubelet.service.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/loadbalancer/haproxy.cfg.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/loadbalancer/nginx.conf.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/manifests/haproxy.manifest.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/manifests/kube-vip.manifest.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/manifests/nginx-proxy.manifest.j2 create mode 100644 kubespray/roles/kubernetes/node/templates/node-kubeconfig.yaml.j2 create mode 100644 kubespray/roles/kubernetes/node/vars/fedora.yml create mode 100644 kubespray/roles/kubernetes/node/vars/ubuntu-18.yml create mode 100644 kubespray/roles/kubernetes/node/vars/ubuntu-20.yml create mode 100644 kubespray/roles/kubernetes/node/vars/ubuntu-22.yml create mode 100644 kubespray/roles/kubernetes/preinstall/defaults/main.yml create mode 100644 kubespray/roles/kubernetes/preinstall/files/dhclient_nodnsupdate create mode 100755 kubespray/roles/kubernetes/preinstall/gen-gitinfos.sh create mode 100644 kubespray/roles/kubernetes/preinstall/handlers/main.yml create mode 100644 kubespray/roles/kubernetes/preinstall/meta/main.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0010-swapoff.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0020-set_facts.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0040-verify-settings.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0050-create_directories.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0060-resolvconf.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0061-systemd-resolved.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0062-networkmanager-unmanaged-devices.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0063-networkmanager-dns.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0070-system-packages.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0080-system-configurations.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0081-ntp-configurations.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0090-etchosts.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0100-dhclient-hooks.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0110-dhclient-hooks-undo.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/0120-growpart-azure-centos-7.yml create mode 100644 kubespray/roles/kubernetes/preinstall/tasks/main.yml create mode 100644 kubespray/roles/kubernetes/preinstall/templates/ansible_git.j2 create mode 100644 kubespray/roles/kubernetes/preinstall/templates/chrony.conf.j2 create mode 100644 kubespray/roles/kubernetes/preinstall/templates/dhclient_dnsupdate.sh.j2 create mode 100644 kubespray/roles/kubernetes/preinstall/templates/dhclient_dnsupdate_rh.sh.j2 create mode 100644 kubespray/roles/kubernetes/preinstall/templates/ntp.conf.j2 create mode 100644 kubespray/roles/kubernetes/preinstall/templates/resolvconf.j2 create mode 100644 kubespray/roles/kubernetes/preinstall/templates/resolved.conf.j2 create mode 100644 kubespray/roles/kubernetes/preinstall/vars/amazon.yml create mode 100644 kubespray/roles/kubernetes/preinstall/vars/centos.yml create mode 100644 kubespray/roles/kubernetes/preinstall/vars/debian-11.yml create mode 100644 kubespray/roles/kubernetes/preinstall/vars/debian-12.yml create mode 100644 kubespray/roles/kubernetes/preinstall/vars/debian.yml create mode 100644 kubespray/roles/kubernetes/preinstall/vars/fedora.yml create mode 100644 kubespray/roles/kubernetes/preinstall/vars/redhat.yml create mode 100644 kubespray/roles/kubernetes/preinstall/vars/suse.yml create mode 100644 kubespray/roles/kubernetes/preinstall/vars/ubuntu.yml create mode 100644 kubespray/roles/kubernetes/tokens/files/kube-gen-token.sh create mode 100644 kubespray/roles/kubernetes/tokens/tasks/check-tokens.yml create mode 100644 kubespray/roles/kubernetes/tokens/tasks/gen_tokens.yml create mode 100644 kubespray/roles/kubernetes/tokens/tasks/main.yml create mode 100644 kubespray/roles/kubespray-defaults/defaults/main.yaml create mode 100644 kubespray/roles/kubespray-defaults/meta/main.yml create mode 100644 kubespray/roles/kubespray-defaults/tasks/fallback_ips.yml create mode 100644 kubespray/roles/kubespray-defaults/tasks/main.yaml create mode 100644 kubespray/roles/kubespray-defaults/tasks/no_proxy.yml create mode 100644 kubespray/roles/kubespray-defaults/vars/main.yml create mode 100755 kubespray/roles/moaroom/defaults/main.yml create mode 100644 kubespray/roles/moaroom/templates/deploy-control.yml.j2 create mode 100644 kubespray/roles/network_plugin/calico/defaults/main.yml create mode 100644 kubespray/roles/network_plugin/calico/files/openssl.conf create mode 100644 kubespray/roles/network_plugin/calico/handlers/main.yml create mode 100644 kubespray/roles/network_plugin/calico/rr/defaults/main.yml create mode 100644 kubespray/roles/network_plugin/calico/rr/tasks/main.yml create mode 100644 kubespray/roles/network_plugin/calico/rr/tasks/pre.yml create mode 100644 kubespray/roles/network_plugin/calico/rr/tasks/update-node.yml create mode 100644 kubespray/roles/network_plugin/calico/tasks/calico_apiserver_certs.yml create mode 100644 kubespray/roles/network_plugin/calico/tasks/check.yml create mode 100644 kubespray/roles/network_plugin/calico/tasks/install.yml create mode 100644 kubespray/roles/network_plugin/calico/tasks/main.yml create mode 100644 kubespray/roles/network_plugin/calico/tasks/peer_with_calico_rr.yml create mode 100644 kubespray/roles/network_plugin/calico/tasks/peer_with_router.yml create mode 100644 kubespray/roles/network_plugin/calico/tasks/pre.yml create mode 100644 kubespray/roles/network_plugin/calico/tasks/repos.yml create mode 100644 kubespray/roles/network_plugin/calico/tasks/reset.yml create mode 100644 kubespray/roles/network_plugin/calico/tasks/typha_certs.yml create mode 100644 kubespray/roles/network_plugin/calico/templates/calico-apiserver-ns.yml.j2 create mode 100644 kubespray/roles/network_plugin/calico/templates/calico-apiserver.yml.j2 create mode 100644 kubespray/roles/network_plugin/calico/templates/calico-config.yml.j2 create mode 100644 kubespray/roles/network_plugin/calico/templates/calico-cr.yml.j2 create mode 100644 kubespray/roles/network_plugin/calico/templates/calico-crb.yml.j2 create mode 100644 kubespray/roles/network_plugin/calico/templates/calico-ipamconfig.yml.j2 create mode 100644 kubespray/roles/network_plugin/calico/templates/calico-node-sa.yml.j2 create mode 100644 kubespray/roles/network_plugin/calico/templates/calico-node.yml.j2 create mode 100644 kubespray/roles/network_plugin/calico/templates/calico-typha.yml.j2 create mode 100644 kubespray/roles/network_plugin/calico/templates/calicoctl.etcd.sh.j2 create mode 100644 kubespray/roles/network_plugin/calico/templates/calicoctl.kdd.sh.j2 create mode 100644 kubespray/roles/network_plugin/calico/templates/kubernetes-services-endpoint.yml.j2 create mode 100644 kubespray/roles/network_plugin/calico/templates/make-ssl-calico.sh.j2 create mode 100644 kubespray/roles/network_plugin/calico/vars/amazon.yml create mode 100644 kubespray/roles/network_plugin/calico/vars/centos-9.yml create mode 100644 kubespray/roles/network_plugin/calico/vars/debian.yml create mode 100644 kubespray/roles/network_plugin/calico/vars/fedora.yml create mode 100644 kubespray/roles/network_plugin/calico/vars/opensuse.yml create mode 100644 kubespray/roles/network_plugin/calico/vars/redhat-9.yml create mode 100644 kubespray/roles/network_plugin/calico/vars/redhat.yml create mode 100644 kubespray/roles/network_plugin/calico/vars/rocky-9.yml create mode 100644 kubespray/roles/network_plugin/cilium/defaults/main.yml create mode 100644 kubespray/roles/network_plugin/cilium/tasks/apply.yml create mode 100644 kubespray/roles/network_plugin/cilium/tasks/check.yml create mode 100644 kubespray/roles/network_plugin/cilium/tasks/install.yml create mode 100644 kubespray/roles/network_plugin/cilium/tasks/main.yml create mode 100644 kubespray/roles/network_plugin/cilium/tasks/reset.yml create mode 100644 kubespray/roles/network_plugin/cilium/tasks/reset_iface.yml create mode 100644 kubespray/roles/network_plugin/cilium/templates/000-cilium-portmap.conflist.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/cilium-operator/cr.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/cilium-operator/crb.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/cilium-operator/deploy.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/cilium-operator/sa.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/cilium/config.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/cilium/cr.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/cilium/crb.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/cilium/ds.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/cilium/sa.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/cilium/secret.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/hubble/config.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/hubble/cr.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/hubble/crb.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/hubble/cronjob.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/hubble/deploy.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/hubble/job.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/hubble/sa.yml.j2 create mode 100644 kubespray/roles/network_plugin/cilium/templates/hubble/service.yml.j2 create mode 100644 kubespray/roles/network_plugin/cni/tasks/main.yml create mode 100644 kubespray/roles/network_plugin/custom_cni/defaults/main.yml create mode 100644 kubespray/roles/network_plugin/custom_cni/tasks/main.yml create mode 100644 kubespray/roles/network_plugin/flannel/defaults/main.yml create mode 100644 kubespray/roles/network_plugin/flannel/meta/main.yml create mode 100644 kubespray/roles/network_plugin/flannel/tasks/main.yml create mode 100644 kubespray/roles/network_plugin/flannel/tasks/reset.yml create mode 100644 kubespray/roles/network_plugin/flannel/templates/cni-flannel-rbac.yml.j2 create mode 100644 kubespray/roles/network_plugin/flannel/templates/cni-flannel.yml.j2 create mode 100644 kubespray/roles/network_plugin/kube-ovn/OWNERS create mode 100644 kubespray/roles/network_plugin/kube-ovn/defaults/main.yml create mode 100644 kubespray/roles/network_plugin/kube-ovn/tasks/main.yml create mode 100644 kubespray/roles/network_plugin/kube-ovn/templates/cni-kube-ovn-crd.yml.j2 create mode 100644 kubespray/roles/network_plugin/kube-ovn/templates/cni-kube-ovn.yml.j2 create mode 100644 kubespray/roles/network_plugin/kube-ovn/templates/cni-ovn.yml.j2 create mode 100644 kubespray/roles/network_plugin/kube-router/OWNERS create mode 100644 kubespray/roles/network_plugin/kube-router/defaults/main.yml create mode 100644 kubespray/roles/network_plugin/kube-router/handlers/main.yml create mode 100644 kubespray/roles/network_plugin/kube-router/meta/main.yml create mode 100644 kubespray/roles/network_plugin/kube-router/tasks/annotate.yml create mode 100644 kubespray/roles/network_plugin/kube-router/tasks/main.yml create mode 100644 kubespray/roles/network_plugin/kube-router/tasks/reset.yml create mode 100644 kubespray/roles/network_plugin/kube-router/templates/cni-conf.json.j2 create mode 100644 kubespray/roles/network_plugin/kube-router/templates/kube-router.yml.j2 create mode 100644 kubespray/roles/network_plugin/kube-router/templates/kubeconfig.yml.j2 create mode 100644 kubespray/roles/network_plugin/macvlan/OWNERS create mode 100644 kubespray/roles/network_plugin/macvlan/defaults/main.yml create mode 100644 kubespray/roles/network_plugin/macvlan/files/ifdown-local create mode 100755 kubespray/roles/network_plugin/macvlan/files/ifdown-macvlan create mode 100755 kubespray/roles/network_plugin/macvlan/files/ifup-local create mode 100755 kubespray/roles/network_plugin/macvlan/files/ifup-macvlan create mode 100644 kubespray/roles/network_plugin/macvlan/handlers/main.yml create mode 100644 kubespray/roles/network_plugin/macvlan/meta/main.yml create mode 100644 kubespray/roles/network_plugin/macvlan/tasks/main.yml create mode 100644 kubespray/roles/network_plugin/macvlan/templates/10-macvlan.conf.j2 create mode 100644 kubespray/roles/network_plugin/macvlan/templates/99-loopback.conf.j2 create mode 100644 kubespray/roles/network_plugin/macvlan/templates/centos-network-macvlan.cfg.j2 create mode 100644 kubespray/roles/network_plugin/macvlan/templates/centos-postdown-macvlan.cfg.j2 create mode 100644 kubespray/roles/network_plugin/macvlan/templates/centos-postup-macvlan.cfg.j2 create mode 100644 kubespray/roles/network_plugin/macvlan/templates/centos-routes-macvlan.cfg.j2 create mode 100644 kubespray/roles/network_plugin/macvlan/templates/coreos-device-macvlan.cfg.j2 create mode 100644 kubespray/roles/network_plugin/macvlan/templates/coreos-interface-macvlan.cfg.j2 create mode 100644 kubespray/roles/network_plugin/macvlan/templates/coreos-network-macvlan.cfg.j2 create mode 100644 kubespray/roles/network_plugin/macvlan/templates/coreos-service-nat_ouside.j2 create mode 100644 kubespray/roles/network_plugin/macvlan/templates/debian-network-macvlan.cfg.j2 create mode 100644 kubespray/roles/network_plugin/meta/main.yml create mode 100644 kubespray/roles/network_plugin/multus/defaults/main.yml create mode 100644 kubespray/roles/network_plugin/multus/files/multus-clusterrole.yml create mode 100644 kubespray/roles/network_plugin/multus/files/multus-clusterrolebinding.yml create mode 100644 kubespray/roles/network_plugin/multus/files/multus-crd.yml create mode 100644 kubespray/roles/network_plugin/multus/files/multus-serviceaccount.yml create mode 100644 kubespray/roles/network_plugin/multus/meta/main.yml create mode 100644 kubespray/roles/network_plugin/multus/tasks/main.yml create mode 100644 kubespray/roles/network_plugin/multus/templates/multus-daemonset.yml.j2 create mode 100644 kubespray/roles/network_plugin/ovn4nfv/tasks/main.yml create mode 100644 kubespray/roles/network_plugin/weave/defaults/main.yml create mode 100644 kubespray/roles/network_plugin/weave/meta/main.yml create mode 100644 kubespray/roles/network_plugin/weave/tasks/main.yml create mode 100644 kubespray/roles/network_plugin/weave/templates/10-weave.conflist.j2 create mode 100644 kubespray/roles/network_plugin/weave/templates/weave-net.yml.j2 create mode 100644 kubespray/roles/recover_control_plane/OWNERS create mode 100644 kubespray/roles/recover_control_plane/control-plane/defaults/main.yml create mode 100644 kubespray/roles/recover_control_plane/control-plane/tasks/main.yml create mode 100644 kubespray/roles/recover_control_plane/etcd/tasks/main.yml create mode 100644 kubespray/roles/recover_control_plane/etcd/tasks/recover_lost_quorum.yml create mode 100644 kubespray/roles/recover_control_plane/post-recover/tasks/main.yml create mode 100644 kubespray/roles/remove-node/post-remove/defaults/main.yml create mode 100644 kubespray/roles/remove-node/post-remove/tasks/main.yml create mode 100644 kubespray/roles/remove-node/pre-remove/defaults/main.yml create mode 100644 kubespray/roles/remove-node/pre-remove/tasks/main.yml create mode 100644 kubespray/roles/remove-node/remove-etcd-node/tasks/main.yml create mode 100644 kubespray/roles/reset/defaults/main.yml create mode 100644 kubespray/roles/reset/tasks/main.yml create mode 100644 kubespray/roles/upgrade/post-upgrade/defaults/main.yml create mode 100644 kubespray/roles/upgrade/post-upgrade/tasks/main.yml create mode 100644 kubespray/roles/upgrade/pre-upgrade/defaults/main.yml create mode 100644 kubespray/roles/upgrade/pre-upgrade/tasks/main.yml create mode 100644 kubespray/roles/upgrade/system-upgrade/tasks/apt.yml create mode 100644 kubespray/roles/upgrade/system-upgrade/tasks/main.yml create mode 100644 kubespray/roles/upgrade/system-upgrade/tasks/yum.yml create mode 100644 kubespray/roles/win_nodes/kubernetes_patch/defaults/main.yml create mode 100644 kubespray/roles/win_nodes/kubernetes_patch/tasks/main.yml create mode 100644 kubespray/run.rc create mode 100644 kubespray/scale.yml create mode 100644 kubespray/scripts/collect-info.yaml create mode 100644 kubespray/scripts/download_hash.py create mode 100755 kubespray/scripts/download_hash.sh create mode 100755 kubespray/scripts/gen_tags.sh create mode 100644 kubespray/scripts/gitlab-branch-cleanup/.gitignore create mode 100644 kubespray/scripts/gitlab-branch-cleanup/README.md create mode 100644 kubespray/scripts/gitlab-branch-cleanup/main.py create mode 100644 kubespray/scripts/gitlab-branch-cleanup/requirements.txt create mode 100644 kubespray/scripts/gitlab-runner.sh create mode 100644 kubespray/scripts/openstack-cleanup/.gitignore create mode 100644 kubespray/scripts/openstack-cleanup/README.md create mode 100755 kubespray/scripts/openstack-cleanup/main.py create mode 100644 kubespray/scripts/openstack-cleanup/requirements.txt create mode 100644 kubespray/scripts/premoderator.sh create mode 100644 kubespray/setup.cfg create mode 100644 kubespray/setup.py create mode 100644 kubespray/test-infra/image-builder/Makefile create mode 100644 kubespray/test-infra/image-builder/OWNERS create mode 100644 kubespray/test-infra/image-builder/cluster.yml create mode 100644 kubespray/test-infra/image-builder/hosts.ini create mode 100644 kubespray/test-infra/image-builder/roles/kubevirt-images/defaults/main.yml create mode 100644 kubespray/test-infra/image-builder/roles/kubevirt-images/tasks/main.yml create mode 100644 kubespray/test-infra/image-builder/roles/kubevirt-images/templates/Dockerfile create mode 100644 kubespray/test-infra/vagrant-docker/Dockerfile create mode 100644 kubespray/test-infra/vagrant-docker/README.md create mode 100755 kubespray/test-infra/vagrant-docker/build.sh create mode 100644 kubespray/tests/Makefile create mode 100644 kubespray/tests/README.md create mode 100644 kubespray/tests/ansible.cfg create mode 100644 kubespray/tests/cloud_playbooks/cleanup-packet.yml create mode 100644 kubespray/tests/cloud_playbooks/create-aws.yml create mode 100644 kubespray/tests/cloud_playbooks/create-do.yml create mode 100644 kubespray/tests/cloud_playbooks/create-gce.yml create mode 100644 kubespray/tests/cloud_playbooks/create-packet.yml create mode 100644 kubespray/tests/cloud_playbooks/delete-aws.yml create mode 100644 kubespray/tests/cloud_playbooks/delete-gce.yml create mode 100644 kubespray/tests/cloud_playbooks/delete-packet.yml create mode 100644 kubespray/tests/cloud_playbooks/roles/cleanup-packet-ci/tasks/main.yml create mode 100644 kubespray/tests/cloud_playbooks/roles/packet-ci/defaults/main.yml create mode 100644 kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/cleanup-old-vms.yml create mode 100644 kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/create-vms.yml create mode 100644 kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/delete-vms.yml create mode 100644 kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/main.yml create mode 100644 kubespray/tests/cloud_playbooks/roles/packet-ci/templates/inventory.j2 create mode 100644 kubespray/tests/cloud_playbooks/roles/packet-ci/templates/vm.yml.j2 create mode 100644 kubespray/tests/cloud_playbooks/templates/boto.j2 create mode 100644 kubespray/tests/cloud_playbooks/templates/gcs_life.json.j2 create mode 100644 kubespray/tests/cloud_playbooks/upload-logs-gcs.yml create mode 100644 kubespray/tests/cloud_playbooks/wait-for-ssh.yml create mode 100644 kubespray/tests/common/_docker_hub_registry_mirror.yml create mode 100644 kubespray/tests/common/_kubespray_test_settings.yml create mode 100644 kubespray/tests/files/custom_cni/README.md create mode 100644 kubespray/tests/files/custom_cni/cilium.yaml create mode 100644 kubespray/tests/files/custom_cni/values.yaml create mode 100644 kubespray/tests/files/packet_almalinux8-calico-ha-ebpf.yml create mode 100644 kubespray/tests/files/packet_almalinux8-calico-nodelocaldns-secondary.yml create mode 100644 kubespray/tests/files/packet_almalinux8-calico-remove-node.yml create mode 100644 kubespray/tests/files/packet_almalinux8-calico.yml create mode 100644 kubespray/tests/files/packet_almalinux8-crio.yml create mode 100644 kubespray/tests/files/packet_almalinux8-docker.yml create mode 100644 kubespray/tests/files/packet_almalinux8-kube-ovn.yml create mode 100644 kubespray/tests/files/packet_amazon-linux-2-aio.yml create mode 100644 kubespray/tests/files/packet_centos7-calico-ha-once-localhost.yml create mode 100644 kubespray/tests/files/packet_centos7-calico-ha.yml create mode 100644 kubespray/tests/files/packet_centos7-flannel-addons-ha.yml create mode 100644 kubespray/tests/files/packet_centos7-multus-calico.yml create mode 100644 kubespray/tests/files/packet_centos7-weave-upgrade-ha.yml create mode 100644 kubespray/tests/files/packet_debian10-calico.yml create mode 100644 kubespray/tests/files/packet_debian10-cilium-svc-proxy.yml create mode 100644 kubespray/tests/files/packet_debian10-docker.yml create mode 100644 kubespray/tests/files/packet_debian10-macvlan.yml create mode 100644 kubespray/tests/files/packet_debian11-calico-upgrade-once.yml create mode 100644 kubespray/tests/files/packet_debian11-calico-upgrade.yml create mode 100644 kubespray/tests/files/packet_debian11-calico.yml create mode 100644 kubespray/tests/files/packet_debian11-custom-cni.yml create mode 100644 kubespray/tests/files/packet_debian11-docker.yml create mode 100644 kubespray/tests/files/packet_debian11-kubelet-csr-approver.yml create mode 100644 kubespray/tests/files/packet_debian12-calico.yml create mode 100644 kubespray/tests/files/packet_debian12-cilium.yml create mode 100644 kubespray/tests/files/packet_debian12-docker.yml create mode 100644 kubespray/tests/files/packet_fedora37-calico-selinux.yml create mode 100644 kubespray/tests/files/packet_fedora37-calico-swap-selinux.yml create mode 100644 kubespray/tests/files/packet_fedora37-crio.yml create mode 100644 kubespray/tests/files/packet_fedora38-docker-calico.yml create mode 100644 kubespray/tests/files/packet_fedora38-docker-weave.yml create mode 100644 kubespray/tests/files/packet_fedora38-kube-ovn.yml create mode 100644 kubespray/tests/files/packet_opensuse-docker-cilium.yml create mode 100644 kubespray/tests/files/packet_rockylinux8-calico.yml create mode 100644 kubespray/tests/files/packet_rockylinux9-calico.yml create mode 100644 kubespray/tests/files/packet_rockylinux9-cilium.yml create mode 100644 kubespray/tests/files/packet_ubuntu20-aio-docker.yml create mode 120000 kubespray/tests/files/packet_ubuntu20-calico-aio-ansible-2_11.yml create mode 100644 kubespray/tests/files/packet_ubuntu20-calico-aio-hardening.yml create mode 100644 kubespray/tests/files/packet_ubuntu20-calico-aio.yml create mode 100644 kubespray/tests/files/packet_ubuntu20-calico-etcd-kubeadm-upgrade-ha.yml create mode 100644 kubespray/tests/files/packet_ubuntu20-calico-etcd-kubeadm.yml create mode 100644 kubespray/tests/files/packet_ubuntu20-calico-ha-recover-noquorum.yml create mode 100644 kubespray/tests/files/packet_ubuntu20-calico-ha-recover.yml create mode 100644 kubespray/tests/files/packet_ubuntu20-calico-ha-wireguard.yml create mode 100644 kubespray/tests/files/packet_ubuntu20-cilium-sep.yml create mode 100644 kubespray/tests/files/packet_ubuntu20-crio.yml create mode 100644 kubespray/tests/files/packet_ubuntu20-docker-weave-sep.yml create mode 100644 kubespray/tests/files/packet_ubuntu20-flannel-ha-once.yml create mode 100644 kubespray/tests/files/packet_ubuntu20-flannel-ha.yml create mode 100644 kubespray/tests/files/packet_ubuntu22-aio-docker.yml create mode 100644 kubespray/tests/files/packet_ubuntu22-calico-aio.yml create mode 100644 kubespray/tests/files/tf-elastx_ubuntu20-calico.yml create mode 100644 kubespray/tests/files/tf-ovh_ubuntu20-calico.yml create mode 100644 kubespray/tests/files/vagrant_centos7-kube-router.rb create mode 100644 kubespray/tests/files/vagrant_centos7-kube-router.yml create mode 100644 kubespray/tests/files/vagrant_fedora37-kube-router.rb create mode 100644 kubespray/tests/files/vagrant_fedora37-kube-router.yml create mode 100644 kubespray/tests/files/vagrant_ubuntu20-calico-dual-stack.rb create mode 100644 kubespray/tests/files/vagrant_ubuntu20-calico-dual-stack.yml create mode 100644 kubespray/tests/files/vagrant_ubuntu20-flannel-collection.rb create mode 100644 kubespray/tests/files/vagrant_ubuntu20-flannel-collection.yml create mode 100644 kubespray/tests/files/vagrant_ubuntu20-flannel.rb create mode 100644 kubespray/tests/files/vagrant_ubuntu20-flannel.yml create mode 100644 kubespray/tests/files/vagrant_ubuntu20-kube-router-sep.rb create mode 100644 kubespray/tests/files/vagrant_ubuntu20-kube-router-sep.yml create mode 100644 kubespray/tests/files/vagrant_ubuntu20-kube-router-svc-proxy.rb create mode 100644 kubespray/tests/files/vagrant_ubuntu20-kube-router-svc-proxy.yml create mode 100644 kubespray/tests/files/vagrant_ubuntu20-weave-medium.rb create mode 100644 kubespray/tests/files/vagrant_ubuntu20-weave-medium.yml create mode 100644 kubespray/tests/local_inventory/host_vars/localhost create mode 100644 kubespray/tests/local_inventory/hosts.cfg create mode 100644 kubespray/tests/requirements.txt create mode 100644 kubespray/tests/requirements.yml create mode 100755 kubespray/tests/run-tests.sh create mode 100644 kubespray/tests/scripts/ansibl8s_test.sh create mode 100755 kubespray/tests/scripts/check_galaxy_version.sh create mode 100755 kubespray/tests/scripts/check_readme_versions.sh create mode 100755 kubespray/tests/scripts/check_typo.sh create mode 100755 kubespray/tests/scripts/create-tf.sh create mode 100755 kubespray/tests/scripts/delete-tf.sh create mode 100755 kubespray/tests/scripts/md-table/main.py create mode 100644 kubespray/tests/scripts/md-table/requirements.txt create mode 100644 kubespray/tests/scripts/md-table/table.md.j2 create mode 100755 kubespray/tests/scripts/md-table/test.sh create mode 100755 kubespray/tests/scripts/molecule_logs.sh create mode 100755 kubespray/tests/scripts/molecule_run.sh create mode 100755 kubespray/tests/scripts/rebase.sh create mode 100755 kubespray/tests/scripts/terraform_install.sh create mode 100755 kubespray/tests/scripts/testcases_cleanup.sh create mode 100755 kubespray/tests/scripts/testcases_prepare.sh create mode 100755 kubespray/tests/scripts/testcases_run.sh create mode 100755 kubespray/tests/scripts/vagrant-validate.sh create mode 100755 kubespray/tests/scripts/vagrant_clean.sh create mode 100755 kubespray/tests/shebang-unit create mode 100644 kubespray/tests/support/aws.groovy create mode 100644 kubespray/tests/templates/fake_hosts.yml.j2 create mode 100644 kubespray/tests/templates/inventory-aws.j2 create mode 100644 kubespray/tests/templates/inventory-do.j2 create mode 100644 kubespray/tests/templates/inventory-gce.j2 create mode 100644 kubespray/tests/testcases/010_check-apiserver.yml create mode 100644 kubespray/tests/testcases/015_check-nodes-ready.yml create mode 100644 kubespray/tests/testcases/020_check-pods-running.yml create mode 100644 kubespray/tests/testcases/030_check-network.yml create mode 100644 kubespray/tests/testcases/040_check-network-adv.yml create mode 100644 kubespray/tests/testcases/100_check-k8s-conformance.yml create mode 100644 kubespray/tests/testcases/roles/cluster-dump/tasks/main.yml create mode 100644 kubespray/upgrade-cluster.yml diff --git a/kubespray/Dockerfile b/kubespray/Dockerfile new file mode 100644 index 0000000..347c1af --- /dev/null +++ b/kubespray/Dockerfile @@ -0,0 +1,45 @@ +# Use imutable image tags rather than mutable tags (like ubuntu:22.04) +FROM ubuntu:jammy-20230308 +# Some tools like yamllint need this +# Pip needs this as well at the moment to install ansible +# (and potentially other packages) +# See: https://github.com/pypa/pip/issues/10219 +ENV LANG=C.UTF-8 \ + DEBIAN_FRONTEND=noninteractive \ + PYTHONDONTWRITEBYTECODE=1 +WORKDIR /kubespray +COPY *.yml ./ +COPY *.cfg ./ +COPY roles ./roles +COPY contrib ./contrib +COPY inventory ./inventory +COPY library ./library +COPY extra_playbooks ./extra_playbooks +COPY playbooks ./playbooks +COPY plugins ./plugins + +RUN apt update -q \ + && apt install -yq --no-install-recommends \ + curl \ + python3 \ + python3-pip \ + sshpass \ + vim \ + rsync \ + openssh-client \ + && pip install --no-compile --no-cache-dir \ + ansible==7.6.0 \ + ansible-core==2.14.6 \ + cryptography==41.0.1 \ + jinja2==3.1.2 \ + netaddr==0.8.0 \ + jmespath==1.0.1 \ + MarkupSafe==2.1.3 \ + ruamel.yaml==0.17.21 \ + passlib==1.7.4 \ + && KUBE_VERSION=$(sed -n 's/^kube_version: //p' roles/kubespray-defaults/defaults/main.yaml) \ + && curl -L https://dl.k8s.io/release/$KUBE_VERSION/bin/linux/$(dpkg --print-architecture)/kubectl -o /usr/local/bin/kubectl \ + && echo $(curl -L https://dl.k8s.io/release/$KUBE_VERSION/bin/linux/$(dpkg --print-architecture)/kubectl.sha256) /usr/local/bin/kubectl | sha256sum --check \ + && chmod a+x /usr/local/bin/kubectl \ + && rm -rf /var/lib/apt/lists/* /var/log/* \ + && find /usr -type d -name '*__pycache__' -prune -exec rm -rf {} \; diff --git a/kubespray/Makefile b/kubespray/Makefile new file mode 100644 index 0000000..793e763 --- /dev/null +++ b/kubespray/Makefile @@ -0,0 +1,7 @@ +mitogen: + @echo Mitogen support is deprecated. + @echo Please run the following command manually: + @echo ansible-playbook -c local mitogen.yml -vv +clean: + rm -rf dist/ + rm *.retry diff --git a/kubespray/README.md b/kubespray/README.md new file mode 100644 index 0000000..7caa9f3 --- /dev/null +++ b/kubespray/README.md @@ -0,0 +1,283 @@ +# Deploy a Production Ready Kubernetes Cluster + +![Kubernetes Logo](https://raw.githubusercontent.com/kubernetes-sigs/kubespray/master/docs/img/kubernetes-logo.png) + +If you have questions, check the documentation at [kubespray.io](https://kubespray.io) and join us on the [kubernetes slack](https://kubernetes.slack.com), channel **\#kubespray**. +You can get your invite [here](http://slack.k8s.io/) + +- Can be deployed on **[AWS](docs/aws.md), GCE, [Azure](docs/azure.md), [OpenStack](docs/openstack.md), [vSphere](docs/vsphere.md), [Equinix Metal](docs/equinix-metal.md) (bare metal), Oracle Cloud Infrastructure (Experimental), or Baremetal** +- **Highly available** cluster +- **Composable** (Choice of the network plugin for instance) +- Supports most popular **Linux distributions** +- **Continuous integration tests** + +## Quick Start + +Below are several ways to use Kubespray to deploy a Kubernetes cluster. + +### Ansible + +#### Usage + +Install Ansible according to [Ansible installation guide](/docs/ansible.md#installing-ansible) +then run the following steps: + +```ShellSession +# Copy ``inventory/sample`` as ``inventory/mycluster`` +cp -rfp inventory/sample inventory/mycluster + +# Update Ansible inventory file with inventory builder +declare -a IPS=(10.10.1.3 10.10.1.4 10.10.1.5) +CONFIG_FILE=inventory/mycluster/hosts.yaml python3 contrib/inventory_builder/inventory.py ${IPS[@]} + +# Review and change parameters under ``inventory/mycluster/group_vars`` +cat inventory/mycluster/group_vars/all/all.yml +cat inventory/mycluster/group_vars/k8s_cluster/k8s-cluster.yml + +# Clean up old Kubernetes cluster with Ansible Playbook - run the playbook as root +# The option `--become` is required, as for example cleaning up SSL keys in /etc/, +# uninstalling old packages and interacting with various systemd daemons. +# Without --become the playbook will fail to run! +# And be mind it will remove the current kubernetes cluster (if it's running)! +ansible-playbook -i inventory/mycluster/hosts.yaml --become --become-user=root reset.yml + +# Deploy Kubespray with Ansible Playbook - run the playbook as root +# The option `--become` is required, as for example writing SSL keys in /etc/, +# installing packages and interacting with various systemd daemons. +# Without --become the playbook will fail to run! +ansible-playbook -i inventory/mycluster/hosts.yaml --become --become-user=root cluster.yml +``` + +Note: When Ansible is already installed via system packages on the control node, +Python packages installed via `sudo pip install -r requirements.txt` will go to +a different directory tree (e.g. `/usr/local/lib/python2.7/dist-packages` on +Ubuntu) from Ansible's (e.g. `/usr/lib/python2.7/dist-packages/ansible` still on +Ubuntu). As a consequence, the `ansible-playbook` command will fail with: + +```raw +ERROR! no action detected in task. This often indicates a misspelled module name, or incorrect module path. +``` + +This likely indicates that a task depends on a module present in ``requirements.txt``. + +One way of addressing this is to uninstall the system Ansible package then +reinstall Ansible via ``pip``, but this not always possible and one must +take care regarding package versions. +A workaround consists of setting the `ANSIBLE_LIBRARY` +and `ANSIBLE_MODULE_UTILS` environment variables respectively to +the `ansible/modules` and `ansible/module_utils` subdirectories of the ``pip`` +installation location, which is the ``Location`` shown by running +`pip show [package]` before executing `ansible-playbook`. + +A simple way to ensure you get all the correct version of Ansible is to use +the [pre-built docker image from Quay](https://quay.io/repository/kubespray/kubespray?tab=tags). +You will then need to use [bind mounts](https://docs.docker.com/storage/bind-mounts/) +to access the inventory and SSH key in the container, like this: + +```ShellSession +git checkout v2.23.0 +docker pull quay.io/kubespray/kubespray:v2.23.0 +docker run --rm -it --mount type=bind,source="$(pwd)"/inventory/sample,dst=/inventory \ + --mount type=bind,source="${HOME}"/.ssh/id_rsa,dst=/root/.ssh/id_rsa \ + quay.io/kubespray/kubespray:v2.23.0 bash +# Inside the container you may now run the kubespray playbooks: +ansible-playbook -i /inventory/inventory.ini --private-key /root/.ssh/id_rsa cluster.yml +``` + +#### Collection + +See [here](docs/ansible_collection.md) if you wish to use this repository as an Ansible collection + +### Vagrant + +For Vagrant we need to install Python dependencies for provisioning tasks. +Check that ``Python`` and ``pip`` are installed: + +```ShellSession +python -V && pip -V +``` + +If this returns the version of the software, you're good to go. If not, download and install Python from here + +Install Ansible according to [Ansible installation guide](/docs/ansible.md#installing-ansible) +then run the following step: + +```ShellSession +vagrant up +``` + +## Documents + +- [Requirements](#requirements) +- [Kubespray vs ...](docs/comparisons.md) +- [Getting started](docs/getting-started.md) +- [Setting up your first cluster](docs/setting-up-your-first-cluster.md) +- [Ansible inventory and tags](docs/ansible.md) +- [Integration with existing ansible repo](docs/integration.md) +- [Deployment data variables](docs/vars.md) +- [DNS stack](docs/dns-stack.md) +- [HA mode](docs/ha-mode.md) +- [Network plugins](#network-plugins) +- [Vagrant install](docs/vagrant.md) +- [Flatcar Container Linux bootstrap](docs/flatcar.md) +- [Fedora CoreOS bootstrap](docs/fcos.md) +- [Debian Jessie setup](docs/debian.md) +- [openSUSE setup](docs/opensuse.md) +- [Downloaded artifacts](docs/downloads.md) +- [Cloud providers](docs/cloud.md) +- [OpenStack](docs/openstack.md) +- [AWS](docs/aws.md) +- [Azure](docs/azure.md) +- [vSphere](docs/vsphere.md) +- [Equinix Metal](docs/equinix-metal.md) +- [Large deployments](docs/large-deployments.md) +- [Adding/replacing a node](docs/nodes.md) +- [Upgrades basics](docs/upgrades.md) +- [Air-Gap installation](docs/offline-environment.md) +- [NTP](docs/ntp.md) +- [Hardening](docs/hardening.md) +- [Mirror](docs/mirror.md) +- [Roadmap](docs/roadmap.md) + +## Supported Linux Distributions + +- **Flatcar Container Linux by Kinvolk** +- **Debian** Bookworm, Bullseye, Buster +- **Ubuntu** 20.04, 22.04 +- **CentOS/RHEL** 7, [8, 9](docs/centos.md#centos-8) +- **Fedora** 37, 38 +- **Fedora CoreOS** (see [fcos Note](docs/fcos.md)) +- **openSUSE** Leap 15.x/Tumbleweed +- **Oracle Linux** 7, [8, 9](docs/centos.md#centos-8) +- **Alma Linux** [8, 9](docs/centos.md#centos-8) +- **Rocky Linux** [8, 9](docs/centos.md#centos-8) +- **Kylin Linux Advanced Server V10** (experimental: see [kylin linux notes](docs/kylinlinux.md)) +- **Amazon Linux 2** (experimental: see [amazon linux notes](docs/amazonlinux.md)) +- **UOS Linux** (experimental: see [uos linux notes](docs/uoslinux.md)) +- **openEuler** (experimental: see [openEuler notes](docs/openeuler.md)) + +Note: Upstart/SysV init based OS types are not supported. + +## Supported Components + +- Core + - [kubernetes](https://github.com/kubernetes/kubernetes) v1.28.2 + - [etcd](https://github.com/etcd-io/etcd) v3.5.9 + - [docker](https://www.docker.com/) v20.10 (see note) + - [containerd](https://containerd.io/) v1.7.6 + - [cri-o](http://cri-o.io/) v1.27 (experimental: see [CRI-O Note](docs/cri-o.md). Only on fedora, ubuntu and centos based OS) +- Network Plugin + - [cni-plugins](https://github.com/containernetworking/plugins) v1.2.0 + - [calico](https://github.com/projectcalico/calico) v3.26.1 + - [cilium](https://github.com/cilium/cilium) v1.13.4 + - [flannel](https://github.com/flannel-io/flannel) v0.22.0 + - [kube-ovn](https://github.com/alauda/kube-ovn) v1.11.5 + - [kube-router](https://github.com/cloudnativelabs/kube-router) v1.6.0 + - [multus](https://github.com/k8snetworkplumbingwg/multus-cni) v3.8 + - [weave](https://github.com/weaveworks/weave) v2.8.1 + - [kube-vip](https://github.com/kube-vip/kube-vip) v0.5.12 +- Application + - [cert-manager](https://github.com/jetstack/cert-manager) v1.11.1 + - [coredns](https://github.com/coredns/coredns) v1.10.1 + - [ingress-nginx](https://github.com/kubernetes/ingress-nginx) v1.8.2 + - [krew](https://github.com/kubernetes-sigs/krew) v0.4.4 + - [argocd](https://argoproj.github.io/) v2.8.0 + - [helm](https://helm.sh/) v3.12.3 + - [metallb](https://metallb.universe.tf/) v0.13.9 + - [registry](https://github.com/distribution/distribution) v2.8.1 +- Storage Plugin + - [cephfs-provisioner](https://github.com/kubernetes-incubator/external-storage) v2.1.0-k8s1.11 + - [rbd-provisioner](https://github.com/kubernetes-incubator/external-storage) v2.1.1-k8s1.11 + - [aws-ebs-csi-plugin](https://github.com/kubernetes-sigs/aws-ebs-csi-driver) v0.5.0 + - [azure-csi-plugin](https://github.com/kubernetes-sigs/azuredisk-csi-driver) v1.10.0 + - [cinder-csi-plugin](https://github.com/kubernetes/cloud-provider-openstack/blob/master/docs/cinder-csi-plugin/using-cinder-csi-plugin.md) v1.22.0 + - [gcp-pd-csi-plugin](https://github.com/kubernetes-sigs/gcp-compute-persistent-disk-csi-driver) v1.9.2 + - [local-path-provisioner](https://github.com/rancher/local-path-provisioner) v0.0.24 + - [local-volume-provisioner](https://github.com/kubernetes-sigs/sig-storage-local-static-provisioner) v2.5.0 + +## Container Runtime Notes + +- Supported Docker versions are 18.09, 19.03, 20.10, 23.0 and 24.0. The *recommended* Docker version is 20.10 (except on Debian bookworm which without supporting for 20.10 and below any more). `Kubelet` might break on docker's non-standard version numbering (it no longer uses semantic versioning). To ensure auto-updates don't break your cluster look into e.g. the YUM ``versionlock`` plugin or ``apt pin``). +- The cri-o version should be aligned with the respective kubernetes version (i.e. kube_version=1.20.x, crio_version=1.20) + +## Requirements + +- **Minimum required version of Kubernetes is v1.26** +- **Ansible v2.14+, Jinja 2.11+ and python-netaddr is installed on the machine that will run Ansible commands** +- The target servers must have **access to the Internet** in order to pull docker images. Otherwise, additional configuration is required (See [Offline Environment](docs/offline-environment.md)) +- The target servers are configured to allow **IPv4 forwarding**. +- If using IPv6 for pods and services, the target servers are configured to allow **IPv6 forwarding**. +- The **firewalls are not managed**, you'll need to implement your own rules the way you used to. + in order to avoid any issue during deployment you should disable your firewall. +- If kubespray is run from non-root user account, correct privilege escalation method + should be configured in the target servers. Then the `ansible_become` flag + or command parameters `--become or -b` should be specified. + +Hardware: +These limits are safeguarded by Kubespray. Actual requirements for your workload can differ. For a sizing guide go to the [Building Large Clusters](https://kubernetes.io/docs/setup/cluster-large/#size-of-master-and-master-components) guide. + +- Master + - Memory: 1500 MB +- Node + - Memory: 1024 MB + +## Network Plugins + +You can choose among ten network plugins. (default: `calico`, except Vagrant uses `flannel`) + +- [flannel](docs/flannel.md): gre/vxlan (layer 2) networking. + +- [Calico](https://docs.tigera.io/calico/latest/about/) is a networking and network policy provider. Calico supports a flexible set of networking options + designed to give you the most efficient networking across a range of situations, including non-overlay + and overlay networks, with or without BGP. Calico uses the same engine to enforce network policy for hosts, + pods, and (if using Istio and Envoy) applications at the service mesh layer. + +- [cilium](http://docs.cilium.io/en/latest/): layer 3/4 networking (as well as layer 7 to protect and secure application protocols), supports dynamic insertion of BPF bytecode into the Linux kernel to implement security services, networking and visibility logic. + +- [weave](docs/weave.md): Weave is a lightweight container overlay network that doesn't require an external K/V database cluster. + (Please refer to `weave` [troubleshooting documentation](https://www.weave.works/docs/net/latest/troubleshooting/)). + +- [kube-ovn](docs/kube-ovn.md): Kube-OVN integrates the OVN-based Network Virtualization with Kubernetes. It offers an advanced Container Network Fabric for Enterprises. + +- [kube-router](docs/kube-router.md): Kube-router is a L3 CNI for Kubernetes networking aiming to provide operational + simplicity and high performance: it uses IPVS to provide Kube Services Proxy (if setup to replace kube-proxy), + iptables for network policies, and BGP for ods L3 networking (with optionally BGP peering with out-of-cluster BGP peers). + It can also optionally advertise routes to Kubernetes cluster Pods CIDRs, ClusterIPs, ExternalIPs and LoadBalancerIPs. + +- [macvlan](docs/macvlan.md): Macvlan is a Linux network driver. Pods have their own unique Mac and Ip address, connected directly the physical (layer 2) network. + +- [multus](docs/multus.md): Multus is a meta CNI plugin that provides multiple network interface support to pods. For each interface Multus delegates CNI calls to secondary CNI plugins such as Calico, macvlan, etc. + +- [custom_cni](roles/network-plugin/custom_cni/) : You can specify some manifests that will be applied to the clusters to bring you own CNI and use non-supported ones by Kubespray. + See `tests/files/custom_cni/README.md` and `tests/files/custom_cni/values.yaml`for an example with a CNI provided by a Helm Chart. + +The network plugin to use is defined by the variable `kube_network_plugin`. There is also an +option to leverage built-in cloud provider networking instead. +See also [Network checker](docs/netcheck.md). + +## Ingress Plugins + +- [nginx](https://kubernetes.github.io/ingress-nginx): the NGINX Ingress Controller. + +- [metallb](docs/metallb.md): the MetalLB bare-metal service LoadBalancer provider. + +## Community docs and resources + +- [kubernetes.io/docs/setup/production-environment/tools/kubespray/](https://kubernetes.io/docs/setup/production-environment/tools/kubespray/) +- [kubespray, monitoring and logging](https://github.com/gregbkr/kubernetes-kargo-logging-monitoring) by @gregbkr +- [Deploy Kubernetes w/ Ansible & Terraform](https://rsmitty.github.io/Terraform-Ansible-Kubernetes/) by @rsmitty +- [Deploy a Kubernetes Cluster with Kubespray (video)](https://www.youtube.com/watch?v=CJ5G4GpqDy0) + +## Tools and projects on top of Kubespray + +- [Digital Rebar Provision](https://github.com/digitalrebar/provision/blob/v4/doc/integrations/ansible.rst) +- [Terraform Contrib](https://github.com/kubernetes-sigs/kubespray/tree/master/contrib/terraform) +- [Kubean](https://github.com/kubean-io/kubean) + +## CI Tests + +[![Build graphs](https://gitlab.com/kargo-ci/kubernetes-sigs-kubespray/badges/master/pipeline.svg)](https://gitlab.com/kargo-ci/kubernetes-sigs-kubespray/-/pipelines) + +CI/end-to-end tests sponsored by: [CNCF](https://cncf.io), [Equinix Metal](https://metal.equinix.com/), [OVHcloud](https://www.ovhcloud.com/), [ELASTX](https://elastx.se/). + +See the [test matrix](docs/test_cases.md) for details. diff --git a/kubespray/SECURITY_CONTACTS b/kubespray/SECURITY_CONTACTS new file mode 100644 index 0000000..21703b3 --- /dev/null +++ b/kubespray/SECURITY_CONTACTS @@ -0,0 +1,15 @@ +# Defined below are the security contacts for this repo. +# +# They are the contact point for the Product Security Committee to reach out +# to for triaging and handling of incoming issues. +# +# The below names agree to abide by the +# [Embargo Policy](https://git.k8s.io/security/private-distributors-list.md#embargo-policy) +# and will be removed and replaced if they violate that agreement. +# +# DO NOT REPORT SECURITY VULNERABILITIES DIRECTLY TO THESE NAMES, FOLLOW THE +# INSTRUCTIONS AT https://kubernetes.io/security/ +mattymo +floryut +oomichi +cristicalin diff --git a/kubespray/Vagrantfile b/kubespray/Vagrantfile new file mode 100644 index 0000000..7ba30f3 --- /dev/null +++ b/kubespray/Vagrantfile @@ -0,0 +1,290 @@ +# -*- mode: ruby -*- +# # vi: set ft=ruby : + +# For help on using kubespray with vagrant, check out docs/vagrant.md + +require 'fileutils' + +Vagrant.require_version ">= 2.0.0" + +CONFIG = File.join(File.dirname(__FILE__), ENV['KUBESPRAY_VAGRANT_CONFIG'] || 'vagrant/config.rb') + +FLATCAR_URL_TEMPLATE = "https://%s.release.flatcar-linux.net/amd64-usr/current/flatcar_production_vagrant.json" + +# Uniq disk UUID for libvirt +DISK_UUID = Time.now.utc.to_i + +SUPPORTED_OS = { + "flatcar-stable" => {box: "flatcar-stable", user: "core", box_url: FLATCAR_URL_TEMPLATE % ["stable"]}, + "flatcar-beta" => {box: "flatcar-beta", user: "core", box_url: FLATCAR_URL_TEMPLATE % ["beta"]}, + "flatcar-alpha" => {box: "flatcar-alpha", user: "core", box_url: FLATCAR_URL_TEMPLATE % ["alpha"]}, + "flatcar-edge" => {box: "flatcar-edge", user: "core", box_url: FLATCAR_URL_TEMPLATE % ["edge"]}, + "ubuntu2004" => {box: "generic/ubuntu2004", user: "vagrant"}, + "ubuntu2204" => {box: "generic/ubuntu2204", user: "vagrant"}, + "centos" => {box: "centos/7", user: "vagrant"}, + "centos-bento" => {box: "bento/centos-7.6", user: "vagrant"}, + "centos8" => {box: "centos/8", user: "vagrant"}, + "centos8-bento" => {box: "bento/centos-8", user: "vagrant"}, + "almalinux8" => {box: "almalinux/8", user: "vagrant"}, + "almalinux8-bento" => {box: "bento/almalinux-8", user: "vagrant"}, + "rockylinux8" => {box: "generic/rocky8", user: "vagrant"}, + "fedora37" => {box: "fedora/37-cloud-base", user: "vagrant"}, + "fedora38" => {box: "fedora/38-cloud-base", user: "vagrant"}, + "opensuse" => {box: "opensuse/Leap-15.4.x86_64", user: "vagrant"}, + "opensuse-tumbleweed" => {box: "opensuse/Tumbleweed.x86_64", user: "vagrant"}, + "oraclelinux" => {box: "generic/oracle7", user: "vagrant"}, + "oraclelinux8" => {box: "generic/oracle8", user: "vagrant"}, + "rhel7" => {box: "generic/rhel7", user: "vagrant"}, + "rhel8" => {box: "generic/rhel8", user: "vagrant"}, +} + +if File.exist?(CONFIG) + require CONFIG +end + +# Defaults for config options defined in CONFIG +$num_instances ||= 3 +$instance_name_prefix ||= "k8s" +$vm_gui ||= false +$vm_memory ||= 2048 +$vm_cpus ||= 2 +$shared_folders ||= {} +$forwarded_ports ||= {} +$subnet ||= "172.18.8" +$subnet_ipv6 ||= "fd3c:b398:0698:0756" +$os ||= "ubuntu2004" +$network_plugin ||= "flannel" +# Setting multi_networking to true will install Multus: https://github.com/k8snetworkplumbingwg/multus-cni +$multi_networking ||= "False" +$download_run_once ||= "True" +$download_force_cache ||= "False" +# The first three nodes are etcd servers +$etcd_instances ||= [$num_instances, 3].min +# The first two nodes are kube masters +$kube_master_instances ||= [$num_instances, 2].min +# All nodes are kube nodes +$kube_node_instances ||= $num_instances +# The following only works when using the libvirt provider +$kube_node_instances_with_disks ||= false +$kube_node_instances_with_disks_size ||= "20G" +$kube_node_instances_with_disks_number ||= 2 +$override_disk_size ||= false +$disk_size ||= "20GB" +$local_path_provisioner_enabled ||= "False" +$local_path_provisioner_claim_root ||= "/opt/local-path-provisioner/" +$libvirt_nested ||= false +# boolean or string (e.g. "-vvv") +$ansible_verbosity ||= false +$ansible_tags ||= ENV['VAGRANT_ANSIBLE_TAGS'] || "" + +$playbook ||= "cluster.yml" + +host_vars = {} + +# throw error if os is not supported +if ! SUPPORTED_OS.key?($os) + puts "Unsupported OS: #{$os}" + puts "Supported OS are: #{SUPPORTED_OS.keys.join(', ')}" + exit 1 +end + +$box = SUPPORTED_OS[$os][:box] +# if $inventory is not set, try to use example +$inventory = "inventory/sample" if ! $inventory +$inventory = File.absolute_path($inventory, File.dirname(__FILE__)) + +# if $inventory has a hosts.ini file use it, otherwise copy over +# vars etc to where vagrant expects dynamic inventory to be +if ! File.exist?(File.join(File.dirname($inventory), "hosts.ini")) + $vagrant_ansible = File.join(File.dirname(__FILE__), ".vagrant", "provisioners", "ansible") + FileUtils.mkdir_p($vagrant_ansible) if ! File.exist?($vagrant_ansible) + $vagrant_inventory = File.join($vagrant_ansible,"inventory") + FileUtils.rm_f($vagrant_inventory) + FileUtils.ln_s($inventory, $vagrant_inventory) +end + +if Vagrant.has_plugin?("vagrant-proxyconf") + $no_proxy = ENV['NO_PROXY'] || ENV['no_proxy'] || "127.0.0.1,localhost" + (1..$num_instances).each do |i| + $no_proxy += ",#{$subnet}.#{i+100}" + end +end + +Vagrant.configure("2") do |config| + + config.vm.box = $box + if SUPPORTED_OS[$os].has_key? :box_url + config.vm.box_url = SUPPORTED_OS[$os][:box_url] + end + config.ssh.username = SUPPORTED_OS[$os][:user] + + # plugin conflict + if Vagrant.has_plugin?("vagrant-vbguest") then + config.vbguest.auto_update = false + end + + # always use Vagrants insecure key + config.ssh.insert_key = false + + if ($override_disk_size) + unless Vagrant.has_plugin?("vagrant-disksize") + system "vagrant plugin install vagrant-disksize" + end + config.disksize.size = $disk_size + end + + (1..$num_instances).each do |i| + config.vm.define vm_name = "%s-%01d" % [$instance_name_prefix, i] do |node| + + node.vm.hostname = vm_name + + if Vagrant.has_plugin?("vagrant-proxyconf") + node.proxy.http = ENV['HTTP_PROXY'] || ENV['http_proxy'] || "" + node.proxy.https = ENV['HTTPS_PROXY'] || ENV['https_proxy'] || "" + node.proxy.no_proxy = $no_proxy + end + + ["vmware_fusion", "vmware_workstation"].each do |vmware| + node.vm.provider vmware do |v| + v.vmx['memsize'] = $vm_memory + v.vmx['numvcpus'] = $vm_cpus + end + end + + node.vm.provider :virtualbox do |vb| + vb.memory = $vm_memory + vb.cpus = $vm_cpus + vb.gui = $vm_gui + vb.linked_clone = true + vb.customize ["modifyvm", :id, "--vram", "8"] # ubuntu defaults to 256 MB which is a waste of precious RAM + vb.customize ["modifyvm", :id, "--audio", "none"] + end + + node.vm.provider :libvirt do |lv| + lv.nested = $libvirt_nested + lv.cpu_mode = "host-model" + lv.memory = $vm_memory + lv.cpus = $vm_cpus + lv.default_prefix = 'kubespray' + # Fix kernel panic on fedora 28 + if $os == "fedora" + lv.cpu_mode = "host-passthrough" + end + end + + if $kube_node_instances_with_disks + # Libvirt + driverletters = ('a'..'z').to_a + node.vm.provider :libvirt do |lv| + # always make /dev/sd{a/b/c} so that CI can ensure that + # virtualbox and libvirt will have the same devices to use for OSDs + (1..$kube_node_instances_with_disks_number).each do |d| + lv.storage :file, :device => "hd#{driverletters[d]}", :path => "disk-#{i}-#{d}-#{DISK_UUID}.disk", :size => $kube_node_instances_with_disks_size, :bus => "scsi" + end + end + end + + if $expose_docker_tcp + node.vm.network "forwarded_port", guest: 2375, host: ($expose_docker_tcp + i - 1), auto_correct: true + end + + $forwarded_ports.each do |guest, host| + node.vm.network "forwarded_port", guest: guest, host: host, auto_correct: true + end + + if ["rhel7","rhel8"].include? $os + # Vagrant synced_folder rsync options cannot be used for RHEL boxes as Rsync package cannot + # be installed until the host is registered with a valid Red Hat support subscription + node.vm.synced_folder ".", "/vagrant", disabled: false + $shared_folders.each do |src, dst| + node.vm.synced_folder src, dst + end + else + node.vm.synced_folder ".", "/vagrant", disabled: false, type: "rsync", rsync__args: ['--verbose', '--archive', '--delete', '-z'] , rsync__exclude: ['.git','venv'] + $shared_folders.each do |src, dst| + node.vm.synced_folder src, dst, type: "rsync", rsync__args: ['--verbose', '--archive', '--delete', '-z'] + end + end + + ip = "#{$subnet}.#{i+100}" + node.vm.network :private_network, + :ip => ip, + :libvirt__guest_ipv6 => 'yes', + :libvirt__ipv6_address => "#{$subnet_ipv6}::#{i+100}", + :libvirt__ipv6_prefix => "64", + :libvirt__forward_mode => "none", + :libvirt__dhcp_enabled => false + + # Disable swap for each vm + node.vm.provision "shell", inline: "swapoff -a" + + # ubuntu2004 and ubuntu2204 have IPv6 explicitly disabled. This undoes that. + if ["ubuntu2004", "ubuntu2204"].include? $os + node.vm.provision "shell", inline: "rm -f /etc/modprobe.d/local.conf" + node.vm.provision "shell", inline: "sed -i '/net.ipv6.conf.all.disable_ipv6/d' /etc/sysctl.d/99-sysctl.conf /etc/sysctl.conf" + end + # Hack for fedora37/38 to get the IP address of the second interface + if ["fedora37", "fedora38"].include? $os + config.vm.provision "shell", inline: <<-SHELL + nmcli conn modify 'Wired connection 2' ipv4.addresses $(cat /etc/sysconfig/network-scripts/ifcfg-eth1 | grep IPADDR | cut -d "=" -f2) + nmcli conn modify 'Wired connection 2' ipv4.method manual + service NetworkManager restart + SHELL + end + + # Disable firewalld on oraclelinux/redhat vms + if ["oraclelinux","oraclelinux8","rhel7","rhel8","rockylinux8"].include? $os + node.vm.provision "shell", inline: "systemctl stop firewalld; systemctl disable firewalld" + end + + host_vars[vm_name] = { + "ip": ip, + "flannel_interface": "eth1", + "kube_network_plugin": $network_plugin, + "kube_network_plugin_multus": $multi_networking, + "download_run_once": $download_run_once, + "download_localhost": "False", + "download_cache_dir": ENV['HOME'] + "/kubespray_cache", + # Make kubespray cache even when download_run_once is false + "download_force_cache": $download_force_cache, + # Keeping the cache on the nodes can improve provisioning speed while debugging kubespray + "download_keep_remote_cache": "False", + "docker_rpm_keepcache": "1", + # These two settings will put kubectl and admin.config in $inventory/artifacts + "kubeconfig_localhost": "True", + "kubectl_localhost": "True", + "local_path_provisioner_enabled": "#{$local_path_provisioner_enabled}", + "local_path_provisioner_claim_root": "#{$local_path_provisioner_claim_root}", + "ansible_ssh_user": SUPPORTED_OS[$os][:user] + } + + # Only execute the Ansible provisioner once, when all the machines are up and ready. + # And limit the action to gathering facts, the full playbook is going to be ran by testcases_run.sh + if i == $num_instances + node.vm.provision "ansible" do |ansible| + ansible.playbook = $playbook + ansible.verbose = $ansible_verbosity + $ansible_inventory_path = File.join( $inventory, "hosts.ini") + if File.exist?($ansible_inventory_path) + ansible.inventory_path = $ansible_inventory_path + end + ansible.become = true + ansible.limit = "all,localhost" + ansible.host_key_checking = false + ansible.raw_arguments = ["--forks=#{$num_instances}", "--flush-cache", "-e ansible_become_pass=vagrant"] + ansible.host_vars = host_vars + if $ansible_tags != "" + ansible.tags = [$ansible_tags] + end + ansible.groups = { + "etcd" => ["#{$instance_name_prefix}-[1:#{$etcd_instances}]"], + "kube_control_plane" => ["#{$instance_name_prefix}-[1:#{$kube_master_instances}]"], + "kube_node" => ["#{$instance_name_prefix}-[1:#{$kube_node_instances}]"], + "k8s_cluster:children" => ["kube_control_plane", "kube_node"], + } + end + end + + end + end +end diff --git a/kubespray/ansible.cfg b/kubespray/ansible.cfg new file mode 100644 index 0000000..e28ce32 --- /dev/null +++ b/kubespray/ansible.cfg @@ -0,0 +1,22 @@ +[ssh_connection] +pipelining=True +ansible_ssh_args = -o ControlMaster=auto -o ControlPersist=30m -o ConnectionAttempts=100 -o UserKnownHostsFile=/dev/null +#control_path = ~/.ssh/ansible-%%r@%%h:%%p +[defaults] +# https://github.com/ansible/ansible/issues/56930 (to ignore group names with - and .) +force_valid_group_names = ignore + +host_key_checking=False +gathering = smart +fact_caching = jsonfile +fact_caching_connection = /tmp +fact_caching_timeout = 86400 +stdout_callback = default +display_skipped_hosts = no +library = ./library +callbacks_enabled = profile_tasks,ara_default +roles_path = roles:$VIRTUAL_ENV/usr/local/share/kubespray/roles:$VIRTUAL_ENV/usr/local/share/ansible/roles:/usr/share/kubespray/roles +deprecation_warnings=False +inventory_ignore_extensions = ~, .orig, .bak, .ini, .cfg, .retry, .pyc, .pyo, .creds, .gpg +[inventory] +ignore_patterns = artifacts, credentials diff --git a/kubespray/cluster.yml b/kubespray/cluster.yml new file mode 100644 index 0000000..7190af4 --- /dev/null +++ b/kubespray/cluster.yml @@ -0,0 +1,3 @@ +--- +- name: Install Kubernetes + ansible.builtin.import_playbook: playbooks/cluster.yml diff --git a/kubespray/contrib/aws_iam/kubernetes-master-policy.json b/kubespray/contrib/aws_iam/kubernetes-master-policy.json new file mode 100644 index 0000000..e5cbaea --- /dev/null +++ b/kubespray/contrib/aws_iam/kubernetes-master-policy.json @@ -0,0 +1,27 @@ +{ + "Version": "2012-10-17", + "Statement": [ + { + "Effect": "Allow", + "Action": ["ec2:*"], + "Resource": ["*"] + }, + { + "Effect": "Allow", + "Action": ["elasticloadbalancing:*"], + "Resource": ["*"] + }, + { + "Effect": "Allow", + "Action": ["route53:*"], + "Resource": ["*"] + }, + { + "Effect": "Allow", + "Action": "s3:*", + "Resource": [ + "arn:aws:s3:::kubernetes-*" + ] + } + ] +} diff --git a/kubespray/contrib/aws_iam/kubernetes-master-role.json b/kubespray/contrib/aws_iam/kubernetes-master-role.json new file mode 100644 index 0000000..66d5de1 --- /dev/null +++ b/kubespray/contrib/aws_iam/kubernetes-master-role.json @@ -0,0 +1,10 @@ +{ + "Version": "2012-10-17", + "Statement": [ + { + "Effect": "Allow", + "Principal": { "Service": "ec2.amazonaws.com"}, + "Action": "sts:AssumeRole" + } + ] +} diff --git a/kubespray/contrib/aws_iam/kubernetes-minion-policy.json b/kubespray/contrib/aws_iam/kubernetes-minion-policy.json new file mode 100644 index 0000000..af81e98 --- /dev/null +++ b/kubespray/contrib/aws_iam/kubernetes-minion-policy.json @@ -0,0 +1,45 @@ +{ + "Version": "2012-10-17", + "Statement": [ + { + "Effect": "Allow", + "Action": "s3:*", + "Resource": [ + "arn:aws:s3:::kubernetes-*" + ] + }, + { + "Effect": "Allow", + "Action": "ec2:Describe*", + "Resource": "*" + }, + { + "Effect": "Allow", + "Action": "ec2:AttachVolume", + "Resource": "*" + }, + { + "Effect": "Allow", + "Action": "ec2:DetachVolume", + "Resource": "*" + }, + { + "Effect": "Allow", + "Action": ["route53:*"], + "Resource": ["*"] + }, + { + "Effect": "Allow", + "Action": [ + "ecr:GetAuthorizationToken", + "ecr:BatchCheckLayerAvailability", + "ecr:GetDownloadUrlForLayer", + "ecr:GetRepositoryPolicy", + "ecr:DescribeRepositories", + "ecr:ListImages", + "ecr:BatchGetImage" + ], + "Resource": "*" + } + ] +} diff --git a/kubespray/contrib/aws_iam/kubernetes-minion-role.json b/kubespray/contrib/aws_iam/kubernetes-minion-role.json new file mode 100644 index 0000000..66d5de1 --- /dev/null +++ b/kubespray/contrib/aws_iam/kubernetes-minion-role.json @@ -0,0 +1,10 @@ +{ + "Version": "2012-10-17", + "Statement": [ + { + "Effect": "Allow", + "Principal": { "Service": "ec2.amazonaws.com"}, + "Action": "sts:AssumeRole" + } + ] +} diff --git a/kubespray/contrib/aws_inventory/kubespray-aws-inventory.py b/kubespray/contrib/aws_inventory/kubespray-aws-inventory.py new file mode 100755 index 0000000..7527c68 --- /dev/null +++ b/kubespray/contrib/aws_inventory/kubespray-aws-inventory.py @@ -0,0 +1,81 @@ +#!/usr/bin/env python + +from __future__ import print_function +import boto3 +import os +import argparse +import json + +class SearchEC2Tags(object): + + def __init__(self): + self.parse_args() + if self.args.list: + self.search_tags() + if self.args.host: + data = {} + print(json.dumps(data, indent=2)) + + def parse_args(self): + + ##Check if VPC_VISIBILITY is set, if not default to private + if "VPC_VISIBILITY" in os.environ: + self.vpc_visibility = os.environ['VPC_VISIBILITY'] + else: + self.vpc_visibility = "private" + + ##Support --list and --host flags. We largely ignore the host one. + parser = argparse.ArgumentParser() + parser.add_argument('--list', action='store_true', default=False, help='List instances') + parser.add_argument('--host', action='store_true', help='Get all the variables about a specific instance') + self.args = parser.parse_args() + + def search_tags(self): + hosts = {} + hosts['_meta'] = { 'hostvars': {} } + + ##Search ec2 three times to find nodes of each group type. Relies on kubespray-role key/value. + for group in ["kube_control_plane", "kube_node", "etcd"]: + hosts[group] = [] + tag_key = "kubespray-role" + tag_value = ["*"+group+"*"] + region = os.environ['AWS_REGION'] + + ec2 = boto3.resource('ec2', region) + filters = [{'Name': 'tag:'+tag_key, 'Values': tag_value}, {'Name': 'instance-state-name', 'Values': ['running']}] + cluster_name = os.getenv('CLUSTER_NAME') + if cluster_name: + filters.append({'Name': 'tag-key', 'Values': ['kubernetes.io/cluster/'+cluster_name]}) + instances = ec2.instances.filter(Filters=filters) + for instance in instances: + + ##Suppose default vpc_visibility is private + dns_name = instance.private_dns_name + ansible_host = { + 'ansible_ssh_host': instance.private_ip_address + } + + ##Override when vpc_visibility actually is public + if self.vpc_visibility == "public": + dns_name = instance.public_dns_name + ansible_host = { + 'ansible_ssh_host': instance.public_ip_address + } + + ##Set when instance actually has node_labels + node_labels_tag = list(filter(lambda t: t['Key'] == 'kubespray-node-labels', instance.tags)) + if node_labels_tag: + ansible_host['node_labels'] = dict([ label.strip().split('=') for label in node_labels_tag[0]['Value'].split(',') ]) + + ##Set when instance actually has node_taints + node_taints_tag = list(filter(lambda t: t['Key'] == 'kubespray-node-taints', instance.tags)) + if node_taints_tag: + ansible_host['node_taints'] = list([ taint.strip() for taint in node_taints_tag[0]['Value'].split(',') ]) + + hosts[group].append(dns_name) + hosts['_meta']['hostvars'][dns_name] = ansible_host + + hosts['k8s_cluster'] = {'children':['kube_control_plane', 'kube_node']} + print(json.dumps(hosts, sort_keys=True, indent=2)) + +SearchEC2Tags() diff --git a/kubespray/contrib/aws_inventory/requirements.txt b/kubespray/contrib/aws_inventory/requirements.txt new file mode 100644 index 0000000..179d5de --- /dev/null +++ b/kubespray/contrib/aws_inventory/requirements.txt @@ -0,0 +1 @@ +boto3 # Apache-2.0 diff --git a/kubespray/contrib/azurerm/.gitignore b/kubespray/contrib/azurerm/.gitignore new file mode 100644 index 0000000..3ef07f8 --- /dev/null +++ b/kubespray/contrib/azurerm/.gitignore @@ -0,0 +1,2 @@ +.generated +/inventory diff --git a/kubespray/contrib/azurerm/README.md b/kubespray/contrib/azurerm/README.md new file mode 100644 index 0000000..f24a5ec --- /dev/null +++ b/kubespray/contrib/azurerm/README.md @@ -0,0 +1,67 @@ +# Kubernetes on Azure with Azure Resource Group Templates + +Provision the base infrastructure for a Kubernetes cluster by using [Azure Resource Group Templates](https://docs.microsoft.com/en-us/azure/azure-resource-manager/resource-group-authoring-templates) + +## Status + +This will provision the base infrastructure (vnet, vms, nics, ips, ...) needed for Kubernetes in Azure into the specified +Resource Group. It will not install Kubernetes itself, this has to be done in a later step by yourself (using kubespray of course). + +## Requirements + +- [Install azure-cli](https://docs.microsoft.com/en-us/cli/azure/install-azure-cli?view=azure-cli-latest) +- [Login with azure-cli](https://docs.microsoft.com/en-us/cli/azure/authenticate-azure-cli?view=azure-cli-latest) +- Dedicated Resource Group created in the Azure Portal or through azure-cli + +## Configuration through group_vars/all + +You have to modify at least two variables in group_vars/all. The one is the **cluster_name** variable, it must be globally +unique due to some restrictions in Azure. The other one is the **ssh_public_keys** variable, it must be your ssh public +key to access your azure virtual machines. Most other variables should be self explanatory if you have some basic Kubernetes +experience. + +## Bastion host + +You can enable the use of a Bastion Host by changing **use_bastion** in group_vars/all to **true**. The generated +templates will then include an additional bastion VM which can then be used to connect to the masters and nodes. The option +also removes all public IPs from all other VMs. + +## Generating and applying + +To generate and apply the templates, call: + +```shell +./apply-rg.sh +``` + +If you change something in the configuration (e.g. number of nodes) later, you can call this again and Azure will +take care about creating/modifying whatever is needed. + +## Clearing a resource group + +If you need to delete all resources from a resource group, simply call: + +```shell +./clear-rg.sh +``` + +**WARNING** this really deletes everything from your resource group, including everything that was later created by you! + +## Installing Ansible and the dependencies + +Install Ansible according to [Ansible installation guide](/docs/ansible.md#installing-ansible) + +## Generating an inventory for kubespray + +After you have applied the templates, you can generate an inventory with this call: + +```shell +./generate-inventory.sh +``` + +It will create the file ./inventory which can then be used with kubespray, e.g.: + +```shell +cd kubespray-root-dir +ansible-playbook -i contrib/azurerm/inventory -u devops --become -e "@inventory/sample/group_vars/all/all.yml" cluster.yml +``` diff --git a/kubespray/contrib/azurerm/apply-rg.sh b/kubespray/contrib/azurerm/apply-rg.sh new file mode 100755 index 0000000..2348169 --- /dev/null +++ b/kubespray/contrib/azurerm/apply-rg.sh @@ -0,0 +1,19 @@ +#!/usr/bin/env bash + +set -e + +AZURE_RESOURCE_GROUP="$1" + +if [ "$AZURE_RESOURCE_GROUP" == "" ]; then + echo "AZURE_RESOURCE_GROUP is missing" + exit 1 +fi + +ansible-playbook generate-templates.yml + +az deployment group create --template-file ./.generated/network.json -g $AZURE_RESOURCE_GROUP +az deployment group create --template-file ./.generated/storage.json -g $AZURE_RESOURCE_GROUP +az deployment group create --template-file ./.generated/availability-sets.json -g $AZURE_RESOURCE_GROUP +az deployment group create --template-file ./.generated/bastion.json -g $AZURE_RESOURCE_GROUP +az deployment group create --template-file ./.generated/masters.json -g $AZURE_RESOURCE_GROUP +az deployment group create --template-file ./.generated/minions.json -g $AZURE_RESOURCE_GROUP diff --git a/kubespray/contrib/azurerm/clear-rg.sh b/kubespray/contrib/azurerm/clear-rg.sh new file mode 100755 index 0000000..a200455 --- /dev/null +++ b/kubespray/contrib/azurerm/clear-rg.sh @@ -0,0 +1,14 @@ +#!/usr/bin/env bash + +set -e + +AZURE_RESOURCE_GROUP="$1" + +if [ "$AZURE_RESOURCE_GROUP" == "" ]; then + echo "AZURE_RESOURCE_GROUP is missing" + exit 1 +fi + +ansible-playbook generate-templates.yml + +az group deployment create -g "$AZURE_RESOURCE_GROUP" --template-file ./.generated/clear-rg.json --mode Complete diff --git a/kubespray/contrib/azurerm/generate-inventory.sh b/kubespray/contrib/azurerm/generate-inventory.sh new file mode 100755 index 0000000..b3eb9c0 --- /dev/null +++ b/kubespray/contrib/azurerm/generate-inventory.sh @@ -0,0 +1,18 @@ +#!/usr/bin/env bash + +set -e + +AZURE_RESOURCE_GROUP="$1" + +if [ "$AZURE_RESOURCE_GROUP" == "" ]; then + echo "AZURE_RESOURCE_GROUP is missing" + exit 1 +fi +# check if azure cli 2.0 exists else use azure cli 1.0 +if az &>/dev/null; then + ansible-playbook generate-inventory_2.yml -e azure_resource_group="$AZURE_RESOURCE_GROUP" +elif azure &>/dev/null; then + ansible-playbook generate-inventory.yml -e azure_resource_group="$AZURE_RESOURCE_GROUP" +else + echo "Azure cli not found" +fi diff --git a/kubespray/contrib/azurerm/generate-inventory.yml b/kubespray/contrib/azurerm/generate-inventory.yml new file mode 100644 index 0000000..01ee386 --- /dev/null +++ b/kubespray/contrib/azurerm/generate-inventory.yml @@ -0,0 +1,6 @@ +--- +- name: Generate Azure inventory + hosts: localhost + gather_facts: False + roles: + - generate-inventory diff --git a/kubespray/contrib/azurerm/generate-inventory_2.yml b/kubespray/contrib/azurerm/generate-inventory_2.yml new file mode 100644 index 0000000..9173e1d --- /dev/null +++ b/kubespray/contrib/azurerm/generate-inventory_2.yml @@ -0,0 +1,6 @@ +--- +- name: Generate Azure inventory + hosts: localhost + gather_facts: False + roles: + - generate-inventory_2 diff --git a/kubespray/contrib/azurerm/generate-templates.yml b/kubespray/contrib/azurerm/generate-templates.yml new file mode 100644 index 0000000..f1fcb62 --- /dev/null +++ b/kubespray/contrib/azurerm/generate-templates.yml @@ -0,0 +1,6 @@ +--- +- name: Generate Azure templates + hosts: localhost + gather_facts: False + roles: + - generate-templates diff --git a/kubespray/contrib/azurerm/group_vars/all b/kubespray/contrib/azurerm/group_vars/all new file mode 100644 index 0000000..44dc1e3 --- /dev/null +++ b/kubespray/contrib/azurerm/group_vars/all @@ -0,0 +1,51 @@ + +# Due to some Azure limitations (ex:- Storage Account's name must be unique), +# this name must be globally unique - it will be used as a prefix for azure components +cluster_name: example + +# Set this to true if you do not want to have public IPs for your masters and minions. This will provision a bastion +# node that can be used to access the masters and minions +use_bastion: false + +# Set this to a preferred name that will be used as the first part of the dns name for your bastotion host. For example: k8s-bastion..cloudapp.azure.com. +# This is convenient when exceptions have to be configured on a firewall to allow ssh to the given bastion host. +# bastion_domain_prefix: k8s-bastion + +number_of_k8s_masters: 3 +number_of_k8s_nodes: 3 + +masters_vm_size: Standard_A2 +masters_os_disk_size: 1000 + +minions_vm_size: Standard_A2 +minions_os_disk_size: 1000 + +admin_username: devops +admin_password: changeme + +# MAKE SURE TO CHANGE THIS TO YOUR PUBLIC KEY to access your azure machines +ssh_public_keys: + - "ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAABAQDLRzcxbsFDdEibiyXCSdIFh7bKbXso1NqlKjEyPTptf3aBXHEhVil0lJRjGpTlpfTy7PHvXFbXIOCdv9tOmeH1uxWDDeZawgPFV6VSZ1QneCL+8bxzhjiCn8133wBSPZkN8rbFKd9eEUUBfx8ipCblYblF9FcidylwtMt5TeEmXk8yRVkPiCuEYuDplhc2H0f4PsK3pFb5aDVdaDT3VeIypnOQZZoUxHWqm6ThyHrzLJd3SrZf+RROFWW1uInIDf/SZlXojczUYoffxgT1lERfOJCHJXsqbZWugbxQBwqsVsX59+KPxFFo6nV88h3UQr63wbFx52/MXkX4WrCkAHzN ablock-vwfs@dell-lappy" + +# Disable using ssh using password. Change it to false to allow to connect to ssh by password +disablePasswordAuthentication: true + +# Azure CIDRs +azure_vnet_cidr: 10.0.0.0/8 +azure_admin_cidr: 10.241.2.0/24 +azure_masters_cidr: 10.0.4.0/24 +azure_minions_cidr: 10.240.0.0/16 + +# Azure loadbalancer port to use to access your cluster +kube_apiserver_port: 6443 + +# Azure Netwoking and storage naming to use with inventory/all.yml +#azure_virtual_network_name: KubeVNET +#azure_subnet_admin_name: ad-subnet +#azure_subnet_masters_name: master-subnet +#azure_subnet_minions_name: minion-subnet +#azure_route_table_name: routetable +#azure_security_group_name: secgroup + +# Storage types available are: "Standard_LRS","Premium_LRS" +#azure_storage_account_type: Standard_LRS diff --git a/kubespray/contrib/azurerm/roles/generate-inventory/tasks/main.yml b/kubespray/contrib/azurerm/roles/generate-inventory/tasks/main.yml new file mode 100644 index 0000000..3eb121a --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-inventory/tasks/main.yml @@ -0,0 +1,15 @@ +--- + +- name: Query Azure VMs + command: azure vm list-ip-address --json {{ azure_resource_group }} + register: vm_list_cmd + +- name: Set vm_list + set_fact: + vm_list: "{{ vm_list_cmd.stdout }}" + +- name: Generate inventory + template: + src: inventory.j2 + dest: "{{ playbook_dir }}/inventory" + mode: 0644 diff --git a/kubespray/contrib/azurerm/roles/generate-inventory/templates/inventory.j2 b/kubespray/contrib/azurerm/roles/generate-inventory/templates/inventory.j2 new file mode 100644 index 0000000..6c5feb2 --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-inventory/templates/inventory.j2 @@ -0,0 +1,33 @@ + +{% for vm in vm_list %} +{% if not use_bastion or vm.name == 'bastion' %} +{{ vm.name }} ansible_ssh_host={{ vm.networkProfile.networkInterfaces[0].expanded.ipConfigurations[0].publicIPAddress.expanded.ipAddress }} ip={{ vm.networkProfile.networkInterfaces[0].expanded.ipConfigurations[0].privateIPAddress }} +{% else %} +{{ vm.name }} ansible_ssh_host={{ vm.networkProfile.networkInterfaces[0].expanded.ipConfigurations[0].privateIPAddress }} +{% endif %} +{% endfor %} + +[kube_control_plane] +{% for vm in vm_list %} +{% if 'kube_control_plane' in vm.tags.roles %} +{{ vm.name }} +{% endif %} +{% endfor %} + +[etcd] +{% for vm in vm_list %} +{% if 'etcd' in vm.tags.roles %} +{{ vm.name }} +{% endif %} +{% endfor %} + +[kube_node] +{% for vm in vm_list %} +{% if 'kube_node' in vm.tags.roles %} +{{ vm.name }} +{% endif %} +{% endfor %} + +[k8s_cluster:children] +kube_node +kube_control_plane diff --git a/kubespray/contrib/azurerm/roles/generate-inventory_2/tasks/main.yml b/kubespray/contrib/azurerm/roles/generate-inventory_2/tasks/main.yml new file mode 100644 index 0000000..c628154 --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-inventory_2/tasks/main.yml @@ -0,0 +1,31 @@ +--- + +- name: Query Azure VMs IPs + command: az vm list-ip-addresses -o json --resource-group {{ azure_resource_group }} + register: vm_ip_list_cmd + +- name: Query Azure VMs Roles + command: az vm list -o json --resource-group {{ azure_resource_group }} + register: vm_list_cmd + +- name: Query Azure Load Balancer Public IP + command: az network public-ip show -o json -g {{ azure_resource_group }} -n kubernetes-api-pubip + register: lb_pubip_cmd + +- name: Set VM IP, roles lists and load balancer public IP + set_fact: + vm_ip_list: "{{ vm_ip_list_cmd.stdout }}" + vm_roles_list: "{{ vm_list_cmd.stdout }}" + lb_pubip: "{{ lb_pubip_cmd.stdout }}" + +- name: Generate inventory + template: + src: inventory.j2 + dest: "{{ playbook_dir }}/inventory" + mode: 0644 + +- name: Generate Load Balancer variables + template: + src: loadbalancer_vars.j2 + dest: "{{ playbook_dir }}/loadbalancer_vars.yml" + mode: 0644 diff --git a/kubespray/contrib/azurerm/roles/generate-inventory_2/templates/inventory.j2 b/kubespray/contrib/azurerm/roles/generate-inventory_2/templates/inventory.j2 new file mode 100644 index 0000000..2f6ac5c --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-inventory_2/templates/inventory.j2 @@ -0,0 +1,33 @@ + +{% for vm in vm_ip_list %} +{% if not use_bastion or vm.virtualMachine.name == 'bastion' %} +{{ vm.virtualMachine.name }} ansible_ssh_host={{ vm.virtualMachine.network.publicIpAddresses[0].ipAddress }} ip={{ vm.virtualMachine.network.privateIpAddresses[0] }} +{% else %} +{{ vm.virtualMachine.name }} ansible_ssh_host={{ vm.virtualMachine.network.privateIpAddresses[0] }} +{% endif %} +{% endfor %} + +[kube_control_plane] +{% for vm in vm_roles_list %} +{% if 'kube_control_plane' in vm.tags.roles %} +{{ vm.name }} +{% endif %} +{% endfor %} + +[etcd] +{% for vm in vm_roles_list %} +{% if 'etcd' in vm.tags.roles %} +{{ vm.name }} +{% endif %} +{% endfor %} + +[kube_node] +{% for vm in vm_roles_list %} +{% if 'kube_node' in vm.tags.roles %} +{{ vm.name }} +{% endif %} +{% endfor %} + +[k8s_cluster:children] +kube_node +kube_control_plane diff --git a/kubespray/contrib/azurerm/roles/generate-inventory_2/templates/loadbalancer_vars.j2 b/kubespray/contrib/azurerm/roles/generate-inventory_2/templates/loadbalancer_vars.j2 new file mode 100644 index 0000000..95a62f3 --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-inventory_2/templates/loadbalancer_vars.j2 @@ -0,0 +1,8 @@ +## External LB example config +apiserver_loadbalancer_domain_name: {{ lb_pubip.dnsSettings.fqdn }} +loadbalancer_apiserver: + address: {{ lb_pubip.ipAddress }} + port: 6443 + +## Internal loadbalancers for apiservers +loadbalancer_apiserver_localhost: false diff --git a/kubespray/contrib/azurerm/roles/generate-templates/defaults/main.yml b/kubespray/contrib/azurerm/roles/generate-templates/defaults/main.yml new file mode 100644 index 0000000..ff6b313 --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-templates/defaults/main.yml @@ -0,0 +1,37 @@ +--- +apiVersion: "2015-06-15" + +virtualNetworkName: "{{ azure_virtual_network_name | default('KubeVNET') }}" + +subnetAdminName: "{{ azure_subnet_admin_name | default('ad-subnet') }}" +subnetMastersName: "{{ azure_subnet_masters_name | default('master-subnet') }}" +subnetMinionsName: "{{ azure_subnet_minions_name | default('minion-subnet') }}" + +routeTableName: "{{ azure_route_table_name | default('routetable') }}" +securityGroupName: "{{ azure_security_group_name | default('secgroup') }}" + +nameSuffix: "{{ cluster_name }}" + +availabilitySetMasters: "master-avs" +availabilitySetMinions: "minion-avs" + +faultDomainCount: 3 +updateDomainCount: 10 + +bastionVmSize: Standard_A0 +bastionVMName: bastion +bastionIPAddressName: bastion-pubip + +disablePasswordAuthentication: true + +sshKeyPath: "/home/{{ admin_username }}/.ssh/authorized_keys" + +imageReference: + publisher: "OpenLogic" + offer: "CentOS" + sku: "7.5" + version: "latest" +imageReferenceJson: "{{ imageReference | to_json }}" + +storageAccountName: "sa{{ nameSuffix | replace('-', '') }}" +storageAccountType: "{{ azure_storage_account_type | default('Standard_LRS') }}" diff --git a/kubespray/contrib/azurerm/roles/generate-templates/tasks/main.yml b/kubespray/contrib/azurerm/roles/generate-templates/tasks/main.yml new file mode 100644 index 0000000..294ee96 --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-templates/tasks/main.yml @@ -0,0 +1,25 @@ +--- +- name: Set base_dir + set_fact: + base_dir: "{{ playbook_dir }}/.generated/" + +- name: Create base_dir + file: + path: "{{ base_dir }}" + state: directory + recurse: true + mode: 0755 + +- name: Store json files in base_dir + template: + src: "{{ item }}" + dest: "{{ base_dir }}/{{ item }}" + mode: 0644 + with_items: + - network.json + - storage.json + - availability-sets.json + - bastion.json + - masters.json + - minions.json + - clear-rg.json diff --git a/kubespray/contrib/azurerm/roles/generate-templates/templates/availability-sets.json b/kubespray/contrib/azurerm/roles/generate-templates/templates/availability-sets.json new file mode 100644 index 0000000..78c1547 --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-templates/templates/availability-sets.json @@ -0,0 +1,30 @@ +{ + "$schema": "https://schema.management.azure.com/schemas/2015-01-01/deploymentTemplate.json#", + "contentVersion": "1.0.0.0", + "parameters": { + }, + "variables": { + }, + "resources": [ + { + "type": "Microsoft.Compute/availabilitySets", + "name": "{{availabilitySetMasters}}", + "apiVersion": "{{apiVersion}}", + "location": "[resourceGroup().location]", + "properties": { + "PlatformFaultDomainCount": "{{faultDomainCount}}", + "PlatformUpdateDomainCount": "{{updateDomainCount}}" + } + }, + { + "type": "Microsoft.Compute/availabilitySets", + "name": "{{availabilitySetMinions}}", + "apiVersion": "{{apiVersion}}", + "location": "[resourceGroup().location]", + "properties": { + "PlatformFaultDomainCount": "{{faultDomainCount}}", + "PlatformUpdateDomainCount": "{{updateDomainCount}}" + } + } + ] +} diff --git a/kubespray/contrib/azurerm/roles/generate-templates/templates/bastion.json b/kubespray/contrib/azurerm/roles/generate-templates/templates/bastion.json new file mode 100644 index 0000000..4cf8fc7 --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-templates/templates/bastion.json @@ -0,0 +1,106 @@ +{ + "$schema": "https://schema.management.azure.com/schemas/2015-01-01/deploymentTemplate.json#", + "contentVersion": "1.0.0.0", + "parameters": { + }, + "variables": { + "vnetID": "[resourceId('Microsoft.Network/virtualNetworks', '{{virtualNetworkName}}')]", + "subnetAdminRef": "[concat(variables('vnetID'),'/subnets/', '{{subnetAdminName}}')]" + }, + "resources": [ + {% if use_bastion %} + { + "apiVersion": "{{apiVersion}}", + "type": "Microsoft.Network/publicIPAddresses", + "name": "{{bastionIPAddressName}}", + "location": "[resourceGroup().location]", + "properties": { + "publicIPAllocationMethod": "Static", + "dnsSettings": { + {% if bastion_domain_prefix %} + "domainNameLabel": "{{ bastion_domain_prefix }}" + {% endif %} + } + } + }, + { + "apiVersion": "{{apiVersion}}", + "type": "Microsoft.Network/networkInterfaces", + "name": "{{bastionVMName}}-nic", + "location": "[resourceGroup().location]", + "dependsOn": [ + "[concat('Microsoft.Network/publicIPAddresses/', '{{bastionIPAddressName}}')]" + ], + "properties": { + "ipConfigurations": [ + { + "name": "BastionIpConfig", + "properties": { + "privateIPAllocationMethod": "Dynamic", + "publicIPAddress": { + "id": "[resourceId('Microsoft.Network/publicIPAddresses', '{{bastionIPAddressName}}')]" + }, + "subnet": { + "id": "[variables('subnetAdminRef')]" + } + } + } + ] + } + }, + { + "apiVersion": "{{apiVersion}}", + "type": "Microsoft.Compute/virtualMachines", + "name": "{{bastionVMName}}", + "location": "[resourceGroup().location]", + "dependsOn": [ + "[concat('Microsoft.Network/networkInterfaces/', '{{bastionVMName}}-nic')]" + ], + "tags": { + "roles": "bastion" + }, + "properties": { + "hardwareProfile": { + "vmSize": "{{bastionVmSize}}" + }, + "osProfile": { + "computerName": "{{bastionVMName}}", + "adminUsername": "{{admin_username}}", + "adminPassword": "{{admin_password}}", + "linuxConfiguration": { + "disablePasswordAuthentication": "true", + "ssh": { + "publicKeys": [ + {% for key in ssh_public_keys %} + { + "path": "{{sshKeyPath}}", + "keyData": "{{key}}" + }{% if loop.index < ssh_public_keys | length %},{% endif %} + {% endfor %} + ] + } + } + }, + "storageProfile": { + "imageReference": {{imageReferenceJson}}, + "osDisk": { + "name": "osdisk", + "vhd": { + "uri": "[concat('http://', '{{storageAccountName}}', '.blob.core.windows.net/vhds/', '{{bastionVMName}}', '-osdisk.vhd')]" + }, + "caching": "ReadWrite", + "createOption": "FromImage" + } + }, + "networkProfile": { + "networkInterfaces": [ + { + "id": "[resourceId('Microsoft.Network/networkInterfaces', '{{bastionVMName}}-nic')]" + } + ] + } + } + } + {% endif %} + ] +} diff --git a/kubespray/contrib/azurerm/roles/generate-templates/templates/clear-rg.json b/kubespray/contrib/azurerm/roles/generate-templates/templates/clear-rg.json new file mode 100644 index 0000000..faf31e8 --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-templates/templates/clear-rg.json @@ -0,0 +1,8 @@ +{ + "$schema": "https://schema.management.azure.com/schemas/2015-01-01/deploymentTemplate.json#", + "contentVersion": "1.0.0.0", + "parameters": {}, + "variables": {}, + "resources": [], + "outputs": {} +} diff --git a/kubespray/contrib/azurerm/roles/generate-templates/templates/masters.json b/kubespray/contrib/azurerm/roles/generate-templates/templates/masters.json new file mode 100644 index 0000000..b299383 --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-templates/templates/masters.json @@ -0,0 +1,198 @@ +{ + "$schema": "https://schema.management.azure.com/schemas/2015-01-01/deploymentTemplate.json#", + "contentVersion": "1.0.0.0", + "parameters": { + }, + "variables": { + "lbDomainName": "{{nameSuffix}}-api", + "lbPublicIPAddressName": "kubernetes-api-pubip", + "lbPublicIPAddressType": "Static", + "lbPublicIPAddressID": "[resourceId('Microsoft.Network/publicIPAddresses',variables('lbPublicIPAddressName'))]", + "lbName": "kubernetes-api", + "lbID": "[resourceId('Microsoft.Network/loadBalancers',variables('lbName'))]", + + "vnetID": "[resourceId('Microsoft.Network/virtualNetworks', '{{virtualNetworkName}}')]", + "kubeMastersSubnetRef": "[concat(variables('vnetID'),'/subnets/', '{{subnetMastersName}}')]" + }, + "resources": [ + { + "apiVersion": "{{apiVersion}}", + "type": "Microsoft.Network/publicIPAddresses", + "name": "[variables('lbPublicIPAddressName')]", + "location": "[resourceGroup().location]", + "properties": { + "publicIPAllocationMethod": "[variables('lbPublicIPAddressType')]", + "dnsSettings": { + "domainNameLabel": "[variables('lbDomainName')]" + } + } + }, + { + "apiVersion": "{{apiVersion}}", + "name": "[variables('lbName')]", + "type": "Microsoft.Network/loadBalancers", + "location": "[resourceGroup().location]", + "dependsOn": [ + "[concat('Microsoft.Network/publicIPAddresses/', variables('lbPublicIPAddressName'))]" + ], + "properties": { + "frontendIPConfigurations": [ + { + "name": "kube-api-frontend", + "properties": { + "publicIPAddress": { + "id": "[variables('lbPublicIPAddressID')]" + } + } + } + ], + "backendAddressPools": [ + { + "name": "kube-api-backend" + } + ], + "loadBalancingRules": [ + { + "name": "kube-api", + "properties": { + "frontendIPConfiguration": { + "id": "[concat(variables('lbID'), '/frontendIPConfigurations/kube-api-frontend')]" + }, + "backendAddressPool": { + "id": "[concat(variables('lbID'), '/backendAddressPools/kube-api-backend')]" + }, + "protocol": "tcp", + "frontendPort": "{{kube_apiserver_port}}", + "backendPort": "{{kube_apiserver_port}}", + "enableFloatingIP": false, + "idleTimeoutInMinutes": 5, + "probe": { + "id": "[concat(variables('lbID'), '/probes/kube-api')]" + } + } + } + ], + "probes": [ + { + "name": "kube-api", + "properties": { + "protocol": "tcp", + "port": "{{kube_apiserver_port}}", + "intervalInSeconds": 5, + "numberOfProbes": 2 + } + } + ] + } + }, + {% for i in range(number_of_k8s_masters) %} + {% if not use_bastion %} + { + "apiVersion": "{{apiVersion}}", + "type": "Microsoft.Network/publicIPAddresses", + "name": "master-{{i}}-pubip", + "location": "[resourceGroup().location]", + "properties": { + "publicIPAllocationMethod": "Static" + } + }, + {% endif %} + { + "apiVersion": "{{apiVersion}}", + "type": "Microsoft.Network/networkInterfaces", + "name": "master-{{i}}-nic", + "location": "[resourceGroup().location]", + "dependsOn": [ + {% if not use_bastion %} + "[concat('Microsoft.Network/publicIPAddresses/', 'master-{{i}}-pubip')]", + {% endif %} + "[concat('Microsoft.Network/loadBalancers/', variables('lbName'))]" + ], + "properties": { + "ipConfigurations": [ + { + "name": "MastersIpConfig", + "properties": { + "privateIPAllocationMethod": "Dynamic", + {% if not use_bastion %} + "publicIPAddress": { + "id": "[resourceId('Microsoft.Network/publicIPAddresses', 'master-{{i}}-pubip')]" + }, + {% endif %} + "subnet": { + "id": "[variables('kubeMastersSubnetRef')]" + }, + "loadBalancerBackendAddressPools": [ + { + "id": "[concat(variables('lbID'), '/backendAddressPools/kube-api-backend')]" + } + ] + } + } + ], + "networkSecurityGroup": { + "id": "[resourceId('Microsoft.Network/networkSecurityGroups', '{{securityGroupName}}')]" + }, + "enableIPForwarding": true + } + }, + { + "type": "Microsoft.Compute/virtualMachines", + "name": "master-{{i}}", + "location": "[resourceGroup().location]", + "dependsOn": [ + "[concat('Microsoft.Network/networkInterfaces/', 'master-{{i}}-nic')]" + ], + "tags": { + "roles": "kube_control_plane,etcd" + }, + "apiVersion": "{{apiVersion}}", + "properties": { + "availabilitySet": { + "id": "[resourceId('Microsoft.Compute/availabilitySets', '{{availabilitySetMasters}}')]" + }, + "hardwareProfile": { + "vmSize": "{{masters_vm_size}}" + }, + "osProfile": { + "computerName": "master-{{i}}", + "adminUsername": "{{admin_username}}", + "adminPassword": "{{admin_password}}", + "linuxConfiguration": { + "disablePasswordAuthentication": "{{disablePasswordAuthentication}}", + "ssh": { + "publicKeys": [ + {% for key in ssh_public_keys %} + { + "path": "{{sshKeyPath}}", + "keyData": "{{key}}" + }{% if loop.index < ssh_public_keys | length %},{% endif %} + {% endfor %} + ] + } + } + }, + "storageProfile": { + "imageReference": {{imageReferenceJson}}, + "osDisk": { + "name": "ma{{nameSuffix}}{{i}}", + "vhd": { + "uri": "[concat('http://','{{storageAccountName}}','.blob.core.windows.net/vhds/master-{{i}}.vhd')]" + }, + "caching": "ReadWrite", + "createOption": "FromImage", + "diskSizeGB": "{{masters_os_disk_size}}" + } + }, + "networkProfile": { + "networkInterfaces": [ + { + "id": "[resourceId('Microsoft.Network/networkInterfaces', 'master-{{i}}-nic')]" + } + ] + } + } + } {% if not loop.last %},{% endif %} + {% endfor %} + ] +} diff --git a/kubespray/contrib/azurerm/roles/generate-templates/templates/minions.json b/kubespray/contrib/azurerm/roles/generate-templates/templates/minions.json new file mode 100644 index 0000000..bd0d059 --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-templates/templates/minions.json @@ -0,0 +1,115 @@ +{ + "$schema": "https://schema.management.azure.com/schemas/2015-01-01/deploymentTemplate.json#", + "contentVersion": "1.0.0.0", + "parameters": { + }, + "variables": { + "vnetID": "[resourceId('Microsoft.Network/virtualNetworks', '{{virtualNetworkName}}')]", + "kubeMinionsSubnetRef": "[concat(variables('vnetID'),'/subnets/', '{{subnetMinionsName}}')]" + }, + "resources": [ + {% for i in range(number_of_k8s_nodes) %} + {% if not use_bastion %} + { + "apiVersion": "{{apiVersion}}", + "type": "Microsoft.Network/publicIPAddresses", + "name": "minion-{{i}}-pubip", + "location": "[resourceGroup().location]", + "properties": { + "publicIPAllocationMethod": "Static" + } + }, + {% endif %} + { + "apiVersion": "{{apiVersion}}", + "type": "Microsoft.Network/networkInterfaces", + "name": "minion-{{i}}-nic", + "location": "[resourceGroup().location]", + "dependsOn": [ + {% if not use_bastion %} + "[concat('Microsoft.Network/publicIPAddresses/', 'minion-{{i}}-pubip')]" + {% endif %} + ], + "properties": { + "ipConfigurations": [ + { + "name": "MinionsIpConfig", + "properties": { + "privateIPAllocationMethod": "Dynamic", + {% if not use_bastion %} + "publicIPAddress": { + "id": "[resourceId('Microsoft.Network/publicIPAddresses', 'minion-{{i}}-pubip')]" + }, + {% endif %} + "subnet": { + "id": "[variables('kubeMinionsSubnetRef')]" + } + } + } + ], + "networkSecurityGroup": { + "id": "[resourceId('Microsoft.Network/networkSecurityGroups', '{{securityGroupName}}')]" + }, + "enableIPForwarding": true + } + }, + { + "type": "Microsoft.Compute/virtualMachines", + "name": "minion-{{i}}", + "location": "[resourceGroup().location]", + "dependsOn": [ + "[concat('Microsoft.Network/networkInterfaces/', 'minion-{{i}}-nic')]" + ], + "tags": { + "roles": "kube_node" + }, + "apiVersion": "{{apiVersion}}", + "properties": { + "availabilitySet": { + "id": "[resourceId('Microsoft.Compute/availabilitySets', '{{availabilitySetMinions}}')]" + }, + "hardwareProfile": { + "vmSize": "{{minions_vm_size}}" + }, + "osProfile": { + "computerName": "minion-{{i}}", + "adminUsername": "{{admin_username}}", + "adminPassword": "{{admin_password}}", + "linuxConfiguration": { + "disablePasswordAuthentication": "{{disablePasswordAuthentication}}", + "ssh": { + "publicKeys": [ + {% for key in ssh_public_keys %} + { + "path": "{{sshKeyPath}}", + "keyData": "{{key}}" + }{% if loop.index < ssh_public_keys | length %},{% endif %} + {% endfor %} + ] + } + } + }, + "storageProfile": { + "imageReference": {{imageReferenceJson}}, + "osDisk": { + "name": "mi{{nameSuffix}}{{i}}", + "vhd": { + "uri": "[concat('http://','{{storageAccountName}}','.blob.core.windows.net/vhds/minion-{{i}}.vhd')]" + }, + "caching": "ReadWrite", + "createOption": "FromImage", + "diskSizeGB": "{{minions_os_disk_size}}" + } + }, + "networkProfile": { + "networkInterfaces": [ + { + "id": "[resourceId('Microsoft.Network/networkInterfaces', 'minion-{{i}}-nic')]" + } + ] + } + } + } {% if not loop.last %},{% endif %} + {% endfor %} + ] +} diff --git a/kubespray/contrib/azurerm/roles/generate-templates/templates/network.json b/kubespray/contrib/azurerm/roles/generate-templates/templates/network.json new file mode 100644 index 0000000..763b3db --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-templates/templates/network.json @@ -0,0 +1,109 @@ +{ + "$schema": "https://schema.management.azure.com/schemas/2015-01-01/deploymentTemplate.json#", + "contentVersion": "1.0.0.0", + "parameters": { + }, + "variables": { + }, + "resources": [ + { + "apiVersion": "{{apiVersion}}", + "type": "Microsoft.Network/routeTables", + "name": "{{routeTableName}}", + "location": "[resourceGroup().location]", + "properties": { + "routes": [ + ] + } + }, + { + "type": "Microsoft.Network/virtualNetworks", + "name": "{{virtualNetworkName}}", + "location": "[resourceGroup().location]", + "apiVersion": "{{apiVersion}}", + "dependsOn": [ + "[concat('Microsoft.Network/routeTables/', '{{routeTableName}}')]" + ], + "properties": { + "addressSpace": { + "addressPrefixes": [ + "{{azure_vnet_cidr}}" + ] + }, + "subnets": [ + { + "name": "{{subnetMastersName}}", + "properties": { + "addressPrefix": "{{azure_masters_cidr}}", + "routeTable": { + "id": "[resourceId('Microsoft.Network/routeTables', '{{routeTableName}}')]" + } + } + }, + { + "name": "{{subnetMinionsName}}", + "properties": { + "addressPrefix": "{{azure_minions_cidr}}", + "routeTable": { + "id": "[resourceId('Microsoft.Network/routeTables', '{{routeTableName}}')]" + } + } + } + {% if use_bastion %} + ,{ + "name": "{{subnetAdminName}}", + "properties": { + "addressPrefix": "{{azure_admin_cidr}}", + "routeTable": { + "id": "[resourceId('Microsoft.Network/routeTables', '{{routeTableName}}')]" + } + } + } + {% endif %} + ] + } + }, + { + "apiVersion": "{{apiVersion}}", + "type": "Microsoft.Network/networkSecurityGroups", + "name": "{{securityGroupName}}", + "location": "[resourceGroup().location]", + "properties": { + "securityRules": [ + {% if not use_bastion %} + { + "name": "ssh", + "properties": { + "description": "Allow SSH", + "protocol": "Tcp", + "sourcePortRange": "*", + "destinationPortRange": "22", + "sourceAddressPrefix": "Internet", + "destinationAddressPrefix": "*", + "access": "Allow", + "priority": 100, + "direction": "Inbound" + } + }, + {% endif %} + { + "name": "kube-api", + "properties": { + "description": "Allow secure kube-api", + "protocol": "Tcp", + "sourcePortRange": "*", + "destinationPortRange": "{{kube_apiserver_port}}", + "sourceAddressPrefix": "Internet", + "destinationAddressPrefix": "*", + "access": "Allow", + "priority": 101, + "direction": "Inbound" + } + } + ] + }, + "resources": [], + "dependsOn": [] + } + ] +} diff --git a/kubespray/contrib/azurerm/roles/generate-templates/templates/storage.json b/kubespray/contrib/azurerm/roles/generate-templates/templates/storage.json new file mode 100644 index 0000000..1ed0866 --- /dev/null +++ b/kubespray/contrib/azurerm/roles/generate-templates/templates/storage.json @@ -0,0 +1,19 @@ +{ + "$schema": "https://schema.management.azure.com/schemas/2015-01-01/deploymentTemplate.json#", + "contentVersion": "1.0.0.0", + "parameters": { + }, + "variables": { + }, + "resources": [ + { + "type": "Microsoft.Storage/storageAccounts", + "name": "{{storageAccountName}}", + "location": "[resourceGroup().location]", + "apiVersion": "{{apiVersion}}", + "properties": { + "accountType": "{{storageAccountType}}" + } + } + ] +} diff --git a/kubespray/contrib/dind/README.md b/kubespray/contrib/dind/README.md new file mode 100644 index 0000000..5e72cfc --- /dev/null +++ b/kubespray/contrib/dind/README.md @@ -0,0 +1,177 @@ +# Kubespray DIND experimental setup + +This ansible playbook creates local docker containers +to serve as Kubernetes "nodes", which in turn will run +"normal" Kubernetes docker containers, a mode usually +called DIND (Docker-IN-Docker). + +The playbook has two roles: + +- dind-host: creates the "nodes" as containers in localhost, with + appropriate settings for DIND (privileged, volume mapping for dind + storage, etc). +- dind-cluster: customizes each node container to have required + system packages installed, and some utils (swapoff, lsattr) + symlinked to /bin/true to ease mimicking a real node. + +This playbook has been test with Ubuntu 16.04 as host and ubuntu:16.04 +as docker images (note that dind-cluster has specific customization +for these images). + +The playbook also creates a `/tmp/kubespray.dind.inventory_builder.sh` +helper (wraps up running `contrib/inventory_builder/inventory.py` with +node containers IPs and prefix). + +## Deploying + +See below for a complete successful run: + +1. Create the node containers + +```shell +# From the kubespray root dir +cd contrib/dind +pip install -r requirements.txt + +ansible-playbook -i hosts dind-cluster.yaml + +# Back to kubespray root +cd ../.. +``` + +NOTE: if the playbook run fails with something like below error +message, you may need to specifically set `ansible_python_interpreter`, +see `./hosts` file for an example expanded localhost entry. + +```shell +failed: [localhost] (item=kube-node1) => {"changed": false, "item": "kube-node1", "msg": "Failed to import docker or docker-py - No module named requests.exceptions. Try `pip install docker` or `pip install docker-py` (Python 2.6)"} +``` + +2. Customize kubespray-dind.yaml + +Note that there's coupling between above created node containers +and `kubespray-dind.yaml` settings, in particular regarding selected `node_distro` +(as set in `group_vars/all/all.yaml`), and docker settings. + +```shell +$EDITOR contrib/dind/kubespray-dind.yaml +``` + +3. Prepare the inventory and run the playbook + +```shell +INVENTORY_DIR=inventory/local-dind +mkdir -p ${INVENTORY_DIR} +rm -f ${INVENTORY_DIR}/hosts.ini +CONFIG_FILE=${INVENTORY_DIR}/hosts.ini /tmp/kubespray.dind.inventory_builder.sh + +ansible-playbook --become -e ansible_ssh_user=debian -i ${INVENTORY_DIR}/hosts.ini cluster.yml --extra-vars @contrib/dind/kubespray-dind.yaml +``` + +NOTE: You could also test other distros without editing files by +passing `--extra-vars` as per below commandline, +replacing `DISTRO` by either `debian`, `ubuntu`, `centos`, `fedora`: + +```shell +cd contrib/dind +ansible-playbook -i hosts dind-cluster.yaml --extra-vars node_distro=DISTRO + +cd ../.. +CONFIG_FILE=inventory/local-dind/hosts.ini /tmp/kubespray.dind.inventory_builder.sh +ansible-playbook --become -e ansible_ssh_user=DISTRO -i inventory/local-dind/hosts.ini cluster.yml --extra-vars @contrib/dind/kubespray-dind.yaml --extra-vars bootstrap_os=DISTRO +``` + +## Resulting deployment + +See below to get an idea on how a completed deployment looks like, +from the host where you ran kubespray playbooks. + +### node_distro: debian + +Running from an Ubuntu Xenial host: + +```shell +$ uname -a +Linux ip-xx-xx-xx-xx 4.4.0-1069-aws #79-Ubuntu SMP Mon Sep 24 +15:01:41 UTC 2018 x86_64 x86_64 x86_64 GNU/Linux + +$ docker ps +CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES +1835dd183b75 debian:9.5 "sh -c 'apt-get -qy …" 43 minutes ago Up 43 minutes kube-node5 +30b0af8d2924 debian:9.5 "sh -c 'apt-get -qy …" 43 minutes ago Up 43 minutes kube-node4 +3e0d1510c62f debian:9.5 "sh -c 'apt-get -qy …" 43 minutes ago Up 43 minutes kube-node3 +738993566f94 debian:9.5 "sh -c 'apt-get -qy …" 44 minutes ago Up 44 minutes kube-node2 +c581ef662ed2 debian:9.5 "sh -c 'apt-get -qy …" 44 minutes ago Up 44 minutes kube-node1 + +$ docker exec kube-node1 kubectl get node +NAME STATUS ROLES AGE VERSION +kube-node1 Ready master,node 18m v1.12.1 +kube-node2 Ready master,node 17m v1.12.1 +kube-node3 Ready node 17m v1.12.1 +kube-node4 Ready node 17m v1.12.1 +kube-node5 Ready node 17m v1.12.1 + +$ docker exec kube-node1 kubectl get pod --all-namespaces +NAMESPACE NAME READY STATUS RESTARTS AGE +default netchecker-agent-67489 1/1 Running 0 2m51s +default netchecker-agent-6qq6s 1/1 Running 0 2m51s +default netchecker-agent-fsw92 1/1 Running 0 2m51s +default netchecker-agent-fw6tl 1/1 Running 0 2m51s +default netchecker-agent-hostnet-8f2zb 1/1 Running 0 3m +default netchecker-agent-hostnet-gq7ml 1/1 Running 0 3m +default netchecker-agent-hostnet-jfkgv 1/1 Running 0 3m +default netchecker-agent-hostnet-kwfwx 1/1 Running 0 3m +default netchecker-agent-hostnet-r46nm 1/1 Running 0 3m +default netchecker-agent-lxdrn 1/1 Running 0 2m51s +default netchecker-server-864bd4c897-9vstl 1/1 Running 0 2m40s +default sh-68fcc6db45-qf55h 1/1 Running 1 12m +kube-system coredns-7598f59475-6vknq 1/1 Running 0 14m +kube-system coredns-7598f59475-l5q5x 1/1 Running 0 14m +kube-system kube-apiserver-kube-node1 1/1 Running 0 17m +kube-system kube-apiserver-kube-node2 1/1 Running 0 18m +kube-system kube-controller-manager-kube-node1 1/1 Running 0 18m +kube-system kube-controller-manager-kube-node2 1/1 Running 0 18m +kube-system kube-proxy-5xx9d 1/1 Running 0 17m +kube-system kube-proxy-cdqq4 1/1 Running 0 17m +kube-system kube-proxy-n64ls 1/1 Running 0 17m +kube-system kube-proxy-pswmj 1/1 Running 0 18m +kube-system kube-proxy-x89qw 1/1 Running 0 18m +kube-system kube-scheduler-kube-node1 1/1 Running 4 17m +kube-system kube-scheduler-kube-node2 1/1 Running 4 18m +kube-system kubernetes-dashboard-5db4d9f45f-548rl 1/1 Running 0 14m +kube-system nginx-proxy-kube-node3 1/1 Running 4 17m +kube-system nginx-proxy-kube-node4 1/1 Running 4 17m +kube-system nginx-proxy-kube-node5 1/1 Running 4 17m +kube-system weave-net-42bfr 2/2 Running 0 16m +kube-system weave-net-6gt8m 2/2 Running 0 16m +kube-system weave-net-88nnc 2/2 Running 0 16m +kube-system weave-net-shckr 2/2 Running 0 16m +kube-system weave-net-xr46t 2/2 Running 0 16m + +$ docker exec kube-node1 curl -s http://localhost:31081/api/v1/connectivity_check +{"Message":"All 10 pods successfully reported back to the server","Absent":null,"Outdated":null} +``` + +## Using ./run-test-distros.sh + +You can use `./run-test-distros.sh` to run a set of tests via DIND, +and excerpt from this script, to get an idea: + +```shell +# The SPEC file(s) must have two arrays as e.g. +# DISTROS=(debian centos) +# EXTRAS=( +# 'kube_network_plugin=calico' +# 'kube_network_plugin=flannel' +# 'kube_network_plugin=weave' +# ) +# that will be tested in a "combinatory" way (e.g. from above there'll be +# be 6 test runs), creating a sequenced -nn.out with each output. +# +# Each $EXTRAS element will be whitespace split, and passed as --extra-vars +# to main kubespray ansible-playbook run. +``` + +See e.g. `test-some_distros-most_CNIs.env` and +`test-some_distros-kube_router_combo.env` in particular for a richer +set of CNI specific `--extra-vars` combo. diff --git a/kubespray/contrib/dind/dind-cluster.yaml b/kubespray/contrib/dind/dind-cluster.yaml new file mode 100644 index 0000000..258837d --- /dev/null +++ b/kubespray/contrib/dind/dind-cluster.yaml @@ -0,0 +1,11 @@ +--- +- name: Create nodes as docker containers + hosts: localhost + gather_facts: False + roles: + - { role: dind-host } + +- name: Customize each node containers + hosts: containers + roles: + - { role: dind-cluster } diff --git a/kubespray/contrib/dind/group_vars/all/all.yaml b/kubespray/contrib/dind/group_vars/all/all.yaml new file mode 100644 index 0000000..fd619a0 --- /dev/null +++ b/kubespray/contrib/dind/group_vars/all/all.yaml @@ -0,0 +1,3 @@ +--- +# See distro.yaml for supported node_distro images +node_distro: debian diff --git a/kubespray/contrib/dind/group_vars/all/distro.yaml b/kubespray/contrib/dind/group_vars/all/distro.yaml new file mode 100644 index 0000000..b9c2670 --- /dev/null +++ b/kubespray/contrib/dind/group_vars/all/distro.yaml @@ -0,0 +1,41 @@ +--- +distro_settings: + debian: &DEBIAN + image: "debian:9.5" + user: "debian" + pid1_exe: /lib/systemd/systemd + init: | + sh -c "apt-get -qy update && apt-get -qy install systemd-sysv dbus && exec /sbin/init" + raw_setup: apt-get -qy update && apt-get -qy install dbus python sudo iproute2 + raw_setup_done: test -x /usr/bin/sudo + agetty_svc: getty@* + ssh_service: ssh + extra_packages: [] + ubuntu: + <<: *DEBIAN + image: "ubuntu:16.04" + user: "ubuntu" + init: | + /sbin/init + centos: &CENTOS + image: "centos:7" + user: "centos" + pid1_exe: /usr/lib/systemd/systemd + init: | + /sbin/init + raw_setup: yum -qy install policycoreutils dbus python sudo iproute iptables + raw_setup_done: test -x /usr/bin/sudo + agetty_svc: getty@* serial-getty@* + ssh_service: sshd + extra_packages: [] + fedora: + <<: *CENTOS + image: "fedora:latest" + user: "fedora" + raw_setup: yum -qy install policycoreutils dbus python sudo iproute iptables; mkdir -p /etc/modules-load.d + extra_packages: + - hostname + - procps + - findutils + - kmod + - iputils diff --git a/kubespray/contrib/dind/hosts b/kubespray/contrib/dind/hosts new file mode 100644 index 0000000..356aa26 --- /dev/null +++ b/kubespray/contrib/dind/hosts @@ -0,0 +1,15 @@ +[local] +# If you created a virtualenv for ansible, you may need to specify running the +# python binary from there instead: +#localhost ansible_connection=local ansible_python_interpreter=/home/user/kubespray/.venv/bin/python +localhost ansible_connection=local + +[containers] +kube-node1 +kube-node2 +kube-node3 +kube-node4 +kube-node5 + +[containers:vars] +ansible_connection=docker diff --git a/kubespray/contrib/dind/kubespray-dind.yaml b/kubespray/contrib/dind/kubespray-dind.yaml new file mode 100644 index 0000000..ecfb557 --- /dev/null +++ b/kubespray/contrib/dind/kubespray-dind.yaml @@ -0,0 +1,22 @@ +--- +# kubespray-dind.yaml: minimal kubespray ansible playbook usable for DIND +# See contrib/dind/README.md +kube_api_anonymous_auth: true + +kubelet_fail_swap_on: false + +# Docker nodes need to have been created with same "node_distro: debian" +# at contrib/dind/group_vars/all/all.yaml +bootstrap_os: debian + +docker_version: latest + +docker_storage_options: -s overlay2 --storage-opt overlay2.override_kernel_check=true -g /dind/docker + +dns_mode: coredns + +deploy_netchecker: True +netcheck_agent_image_repo: quay.io/l23network/k8s-netchecker-agent +netcheck_server_image_repo: quay.io/l23network/k8s-netchecker-server +netcheck_agent_image_tag: v1.0 +netcheck_server_image_tag: v1.0 diff --git a/kubespray/contrib/dind/requirements.txt b/kubespray/contrib/dind/requirements.txt new file mode 100644 index 0000000..bdb9670 --- /dev/null +++ b/kubespray/contrib/dind/requirements.txt @@ -0,0 +1 @@ +docker diff --git a/kubespray/contrib/dind/roles/dind-cluster/tasks/main.yaml b/kubespray/contrib/dind/roles/dind-cluster/tasks/main.yaml new file mode 100644 index 0000000..1cf819f --- /dev/null +++ b/kubespray/contrib/dind/roles/dind-cluster/tasks/main.yaml @@ -0,0 +1,73 @@ +--- +- name: Set_fact distro_setup + set_fact: + distro_setup: "{{ distro_settings[node_distro] }}" + +- name: Set_fact other distro settings + set_fact: + distro_user: "{{ distro_setup['user'] }}" + distro_ssh_service: "{{ distro_setup['ssh_service'] }}" + distro_extra_packages: "{{ distro_setup['extra_packages'] }}" + +- name: Null-ify some linux tools to ease DIND + file: + src: "/bin/true" + dest: "{{ item }}" + state: link + force: yes + with_items: + # DIND box may have swap enable, don't bother + - /sbin/swapoff + # /etc/hosts handling would fail on trying to copy file attributes on edit, + # void it by successfully returning nil output + - /usr/bin/lsattr + # disable selinux-isms, sp needed if running on non-Selinux host + - /usr/sbin/semodule + +- name: Void installing dpkg docs and man pages on Debian based distros + copy: + content: | + # Delete locales + path-exclude=/usr/share/locale/* + # Delete man pages + path-exclude=/usr/share/man/* + # Delete docs + path-exclude=/usr/share/doc/* + path-include=/usr/share/doc/*/copyright + dest: /etc/dpkg/dpkg.cfg.d/01_nodoc + mode: 0644 + when: + - ansible_os_family == 'Debian' + +- name: Install system packages to better match a full-fledge node + package: + name: "{{ item }}" + state: present + with_items: "{{ distro_extra_packages + ['rsyslog', 'openssh-server'] }}" + +- name: Start needed services + service: + name: "{{ item }}" + state: started + with_items: + - rsyslog + - "{{ distro_ssh_service }}" + +- name: Create distro user "{{ distro_user }}" + user: + name: "{{ distro_user }}" + uid: 1000 + # groups: sudo + append: yes + +- name: Allow password-less sudo to "{{ distro_user }}" + copy: + content: "{{ distro_user }} ALL=(ALL) NOPASSWD:ALL" + dest: "/etc/sudoers.d/{{ distro_user }}" + mode: 0640 + +- name: "Add my pubkey to {{ distro_user }} user authorized keys" + ansible.posix.authorized_key: + user: "{{ distro_user }}" + state: present + key: "{{ lookup('file', lookup('env', 'HOME') + '/.ssh/id_rsa.pub') }}" diff --git a/kubespray/contrib/dind/roles/dind-host/tasks/main.yaml b/kubespray/contrib/dind/roles/dind-host/tasks/main.yaml new file mode 100644 index 0000000..e44047f --- /dev/null +++ b/kubespray/contrib/dind/roles/dind-host/tasks/main.yaml @@ -0,0 +1,87 @@ +--- +- name: Set_fact distro_setup + set_fact: + distro_setup: "{{ distro_settings[node_distro] }}" + +- name: Set_fact other distro settings + set_fact: + distro_image: "{{ distro_setup['image'] }}" + distro_init: "{{ distro_setup['init'] }}" + distro_pid1_exe: "{{ distro_setup['pid1_exe'] }}" + distro_raw_setup: "{{ distro_setup['raw_setup'] }}" + distro_raw_setup_done: "{{ distro_setup['raw_setup_done'] }}" + distro_agetty_svc: "{{ distro_setup['agetty_svc'] }}" + +- name: Create dind node containers from "containers" inventory section + community.docker.docker_container: + image: "{{ distro_image }}" + name: "{{ item }}" + state: started + hostname: "{{ item }}" + command: "{{ distro_init }}" + # recreate: yes + privileged: true + tmpfs: + - /sys/module/nf_conntrack/parameters + volumes: + - /boot:/boot + - /lib/modules:/lib/modules + - "{{ item }}:/dind/docker" + register: containers + with_items: "{{ groups.containers }}" + tags: + - addresses + +- name: Gather list of containers IPs + set_fact: + addresses: "{{ containers.results | map(attribute='ansible_facts') | map(attribute='docker_container') | map(attribute='NetworkSettings') | map(attribute='IPAddress') | list }}" + tags: + - addresses + +- name: Create inventory_builder helper already set with the list of node containers' IPs + template: + src: inventory_builder.sh.j2 + dest: /tmp/kubespray.dind.inventory_builder.sh + mode: 0755 + tags: + - addresses + +- name: Install needed packages into node containers via raw, need to wait for possible systemd packages to finish installing + raw: | + # agetty processes churn a lot of cpu time failing on inexistent ttys, early STOP them, to rip them in below task + pkill -STOP agetty || true + {{ distro_raw_setup_done }} && echo SKIPPED && exit 0 + until [ "$(readlink /proc/1/exe)" = "{{ distro_pid1_exe }}" ] ; do sleep 1; done + {{ distro_raw_setup }} + delegate_to: "{{ item._ansible_item_label | default(item.item) }}" + with_items: "{{ containers.results }}" + register: result + changed_when: result.stdout.find("SKIPPED") < 0 + +- name: Remove gettys from node containers + raw: | + until test -S /var/run/dbus/system_bus_socket; do sleep 1; done + systemctl disable {{ distro_agetty_svc }} + systemctl stop {{ distro_agetty_svc }} + delegate_to: "{{ item._ansible_item_label | default(item.item) }}" + with_items: "{{ containers.results }}" + changed_when: false + +# Running systemd-machine-id-setup doesn't create a unique id for each node container on Debian, +# handle manually +- name: Re-create unique machine-id (as we may just get what comes in the docker image), needed by some CNIs for mac address seeding (notably weave) + raw: | + echo {{ item | hash('sha1') }} > /etc/machine-id.new + mv -b /etc/machine-id.new /etc/machine-id + cmp /etc/machine-id /etc/machine-id~ || true + systemctl daemon-reload + delegate_to: "{{ item._ansible_item_label | default(item.item) }}" + with_items: "{{ containers.results }}" + +- name: Early hack image install to adapt for DIND + raw: | + rm -fv /usr/bin/udevadm /usr/sbin/udevadm + delegate_to: "{{ item._ansible_item_label | default(item.item) }}" + with_items: "{{ containers.results }}" + register: result + changed_when: result.stdout.find("removed") >= 0 diff --git a/kubespray/contrib/dind/roles/dind-host/templates/inventory_builder.sh.j2 b/kubespray/contrib/dind/roles/dind-host/templates/inventory_builder.sh.j2 new file mode 100644 index 0000000..48e1758 --- /dev/null +++ b/kubespray/contrib/dind/roles/dind-host/templates/inventory_builder.sh.j2 @@ -0,0 +1,3 @@ +#!/bin/bash +# NOTE: if you change HOST_PREFIX, you also need to edit ./hosts [containers] section +HOST_PREFIX=kube-node python3 contrib/inventory_builder/inventory.py {% for ip in addresses %} {{ ip }} {% endfor %} diff --git a/kubespray/contrib/dind/run-test-distros.sh b/kubespray/contrib/dind/run-test-distros.sh new file mode 100755 index 0000000..3695276 --- /dev/null +++ b/kubespray/contrib/dind/run-test-distros.sh @@ -0,0 +1,93 @@ +#!/bin/bash +# Q&D test'em all: creates full DIND kubespray deploys +# for each distro, verifying it via netchecker. + +info() { + local msg="$*" + local date="$(date -Isec)" + echo "INFO: [$date] $msg" +} +pass_or_fail() { + local rc="$?" + local msg="$*" + local date="$(date -Isec)" + [ $rc -eq 0 ] && echo "PASS: [$date] $msg" || echo "FAIL: [$date] $msg" + return $rc +} +test_distro() { + local distro=${1:?};shift + local extra="${*:-}" + local prefix="${distro[${extra}]}" + ansible-playbook -i hosts dind-cluster.yaml -e node_distro=$distro + pass_or_fail "$prefix: dind-nodes" || return 1 + (cd ../.. + INVENTORY_DIR=inventory/local-dind + mkdir -p ${INVENTORY_DIR} + rm -f ${INVENTORY_DIR}/hosts.ini + CONFIG_FILE=${INVENTORY_DIR}/hosts.ini /tmp/kubespray.dind.inventory_builder.sh + # expand $extra with -e in front of each word + extra_args=""; for extra_arg in $extra; do extra_args="$extra_args -e $extra_arg"; done + ansible-playbook --become -e ansible_ssh_user=$distro -i \ + ${INVENTORY_DIR}/hosts.ini cluster.yml \ + -e @contrib/dind/kubespray-dind.yaml -e bootstrap_os=$distro ${extra_args} + pass_or_fail "$prefix: kubespray" + ) || return 1 + local node0=${NODES[0]} + docker exec ${node0} kubectl get pod --all-namespaces + pass_or_fail "$prefix: kube-api" || return 1 + let retries=60 + while ((retries--)); do + # Some CNI may set NodePort on "main" node interface address (thus no localhost NodePort) + # e.g. kube-router: https://github.com/cloudnativelabs/kube-router/pull/217 + docker exec ${node0} curl -m2 -s http://${NETCHECKER_HOST:?}:31081/api/v1/connectivity_check | grep successfully && break + sleep 2 + done + [ $retries -ge 0 ] + pass_or_fail "$prefix: netcheck" || return 1 +} + +NODES=($(egrep ^kube_node hosts)) +NETCHECKER_HOST=localhost + +: ${OUTPUT_DIR:=./out} +mkdir -p ${OUTPUT_DIR} + +# The SPEC file(s) must have two arrays as e.g. +# DISTROS=(debian centos) +# EXTRAS=( +# 'kube_network_plugin=calico' +# 'kube_network_plugin=flannel' +# 'kube_network_plugin=weave' +# ) +# that will be tested in a "combinatory" way (e.g. from above there'll be +# be 6 test runs), creating a sequenced -nn.out with each output. +# +# Each $EXTRAS element will be whitespace split, and passed as --extra-vars +# to main kubespray ansible-playbook run. + +SPECS=${*:?Missing SPEC files, e.g. test-most_distros-some_CNIs.env} +for spec in ${SPECS}; do + unset DISTROS EXTRAS + echo "Loading file=${spec} ..." + . ${spec} || continue + : ${DISTROS:?} || continue + echo "DISTROS:" "${DISTROS[@]}" + echo "EXTRAS->" + printf " %s\n" "${EXTRAS[@]}" + let n=1 + for distro in "${DISTROS[@]}"; do + for extra in "${EXTRAS[@]:-NULL}"; do + # Magic value to let this for run once: + [[ ${extra} == NULL ]] && unset extra + docker rm -f "${NODES[@]}" + printf -v file_out "%s/%s-%02d.out" ${OUTPUT_DIR} ${spec} $((n++)) + { + info "${distro}[${extra}] START: file_out=${file_out}" + time test_distro ${distro} ${extra} + } |& tee ${file_out} + # sleeping for the sake of the human to verify if they want + sleep 2m + done + done +done +egrep -H '^(....:|real)' $(ls -tr ${OUTPUT_DIR}/*.out) diff --git a/kubespray/contrib/dind/test-most_distros-some_CNIs.env b/kubespray/contrib/dind/test-most_distros-some_CNIs.env new file mode 100644 index 0000000..f6e4e1a --- /dev/null +++ b/kubespray/contrib/dind/test-most_distros-some_CNIs.env @@ -0,0 +1,11 @@ +# Test spec file: used from ./run-test-distros.sh, will run +# each distro in $DISTROS overloading main kubespray ansible-playbook run +# Get all DISTROS from distro.yaml (shame no yaml parsing, but nuff anyway) +# DISTROS="${*:-$(egrep -o '^ \w+' group_vars/all/distro.yaml|paste -s)}" +DISTROS=(debian ubuntu centos fedora) + +# Each line below will be added as --extra-vars to main playbook run +EXTRAS=( + 'kube_network_plugin=calico' + 'kube_network_plugin=weave' +) diff --git a/kubespray/contrib/dind/test-some_distros-kube_router_combo.env b/kubespray/contrib/dind/test-some_distros-kube_router_combo.env new file mode 100644 index 0000000..f267712 --- /dev/null +++ b/kubespray/contrib/dind/test-some_distros-kube_router_combo.env @@ -0,0 +1,6 @@ +DISTROS=(debian centos) +NETCHECKER_HOST=${NODES[0]} +EXTRAS=( + 'kube_network_plugin=kube-router {"kube_router_run_service_proxy":false}' + 'kube_network_plugin=kube-router {"kube_router_run_service_proxy":true}' +) diff --git a/kubespray/contrib/dind/test-some_distros-most_CNIs.env b/kubespray/contrib/dind/test-some_distros-most_CNIs.env new file mode 100644 index 0000000..2fb185c --- /dev/null +++ b/kubespray/contrib/dind/test-some_distros-most_CNIs.env @@ -0,0 +1,8 @@ +DISTROS=(debian centos) +EXTRAS=( + 'kube_network_plugin=calico {}' + 'kube_network_plugin=canal {}' + 'kube_network_plugin=cilium {}' + 'kube_network_plugin=flannel {}' + 'kube_network_plugin=weave {}' +) diff --git a/kubespray/contrib/inventory_builder/inventory.py b/kubespray/contrib/inventory_builder/inventory.py new file mode 100644 index 0000000..76e7c0c --- /dev/null +++ b/kubespray/contrib/inventory_builder/inventory.py @@ -0,0 +1,480 @@ +#!/usr/bin/env python3 +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or +# implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# Usage: inventory.py ip1 [ip2 ...] +# Examples: inventory.py 10.10.1.3 10.10.1.4 10.10.1.5 +# +# Advanced usage: +# Add another host after initial creation: inventory.py 10.10.1.5 +# Add range of hosts: inventory.py 10.10.1.3-10.10.1.5 +# Add hosts with different ip and access ip: +# inventory.py 10.0.0.1,192.168.10.1 10.0.0.2,192.168.10.2 10.0.0.3,192.168.1.3 +# Add hosts with a specific hostname, ip, and optional access ip: +# inventory.py first,10.0.0.1,192.168.10.1 second,10.0.0.2 last,10.0.0.3 +# Delete a host: inventory.py -10.10.1.3 +# Delete a host by id: inventory.py -node1 +# +# Load a YAML or JSON file with inventory data: inventory.py load hosts.yaml +# YAML file should be in the following format: +# group1: +# host1: +# ip: X.X.X.X +# var: val +# group2: +# host2: +# ip: X.X.X.X + +from collections import OrderedDict +from ipaddress import ip_address +from ruamel.yaml import YAML + +import os +import re +import subprocess +import sys + +ROLES = ['all', 'kube_control_plane', 'kube_node', 'etcd', 'k8s_cluster', + 'calico_rr'] +PROTECTED_NAMES = ROLES +AVAILABLE_COMMANDS = ['help', 'print_cfg', 'print_ips', 'print_hostnames', + 'load', 'add'] +_boolean_states = {'1': True, 'yes': True, 'true': True, 'on': True, + '0': False, 'no': False, 'false': False, 'off': False} +yaml = YAML() +yaml.Representer.add_representer(OrderedDict, yaml.Representer.represent_dict) + + +def get_var_as_bool(name, default): + value = os.environ.get(name, '') + return _boolean_states.get(value.lower(), default) + +# Configurable as shell vars start + + +CONFIG_FILE = os.environ.get("CONFIG_FILE", "./inventory/sample/hosts.yaml") +# Remove the reference of KUBE_MASTERS after some deprecation cycles. +KUBE_CONTROL_HOSTS = int(os.environ.get("KUBE_CONTROL_HOSTS", + os.environ.get("KUBE_MASTERS", 2))) +# Reconfigures cluster distribution at scale +SCALE_THRESHOLD = int(os.environ.get("SCALE_THRESHOLD", 50)) +MASSIVE_SCALE_THRESHOLD = int(os.environ.get("MASSIVE_SCALE_THRESHOLD", 200)) + +DEBUG = get_var_as_bool("DEBUG", True) +HOST_PREFIX = os.environ.get("HOST_PREFIX", "node") +USE_REAL_HOSTNAME = get_var_as_bool("USE_REAL_HOSTNAME", False) + +# Configurable as shell vars end + + +class KubesprayInventory(object): + + def __init__(self, changed_hosts=None, config_file=None): + self.config_file = config_file + self.yaml_config = {} + loadPreviousConfig = False + printHostnames = False + # See whether there are any commands to process + if changed_hosts and changed_hosts[0] in AVAILABLE_COMMANDS: + if changed_hosts[0] == "add": + loadPreviousConfig = True + changed_hosts = changed_hosts[1:] + elif changed_hosts[0] == "print_hostnames": + loadPreviousConfig = True + printHostnames = True + else: + self.parse_command(changed_hosts[0], changed_hosts[1:]) + sys.exit(0) + + # If the user wants to remove a node, we need to load the config anyway + if changed_hosts and changed_hosts[0][0] == "-": + loadPreviousConfig = True + + if self.config_file and loadPreviousConfig: # Load previous YAML file + try: + self.hosts_file = open(config_file, 'r') + self.yaml_config = yaml.load(self.hosts_file) + except OSError as e: + # I am assuming we are catching "cannot open file" exceptions + print(e) + sys.exit(1) + + if printHostnames: + self.print_hostnames() + sys.exit(0) + + self.ensure_required_groups(ROLES) + + if changed_hosts: + changed_hosts = self.range2ips(changed_hosts) + self.hosts = self.build_hostnames(changed_hosts, + loadPreviousConfig) + self.purge_invalid_hosts(self.hosts.keys(), PROTECTED_NAMES) + self.set_all(self.hosts) + self.set_k8s_cluster() + etcd_hosts_count = 3 if len(self.hosts.keys()) >= 3 else 1 + self.set_etcd(list(self.hosts.keys())[:etcd_hosts_count]) + if len(self.hosts) >= SCALE_THRESHOLD: + self.set_kube_control_plane(list(self.hosts.keys())[ + etcd_hosts_count:(etcd_hosts_count + KUBE_CONTROL_HOSTS)]) + else: + self.set_kube_control_plane( + list(self.hosts.keys())[:KUBE_CONTROL_HOSTS]) + self.set_kube_node(self.hosts.keys()) + if len(self.hosts) >= SCALE_THRESHOLD: + self.set_calico_rr(list(self.hosts.keys())[:etcd_hosts_count]) + else: # Show help if no options + self.show_help() + sys.exit(0) + + self.write_config(self.config_file) + + def write_config(self, config_file): + if config_file: + with open(self.config_file, 'w') as f: + yaml.dump(self.yaml_config, f) + + else: + print("WARNING: Unable to save config. Make sure you set " + "CONFIG_FILE env var.") + + def debug(self, msg): + if DEBUG: + print("DEBUG: {0}".format(msg)) + + def get_ip_from_opts(self, optstring): + if 'ip' in optstring: + return optstring['ip'] + else: + raise ValueError("IP parameter not found in options") + + def ensure_required_groups(self, groups): + for group in groups: + if group == 'all': + self.debug("Adding group {0}".format(group)) + if group not in self.yaml_config: + all_dict = OrderedDict([('hosts', OrderedDict({})), + ('children', OrderedDict({}))]) + self.yaml_config = {'all': all_dict} + else: + self.debug("Adding group {0}".format(group)) + if group not in self.yaml_config['all']['children']: + self.yaml_config['all']['children'][group] = {'hosts': {}} + + def get_host_id(self, host): + '''Returns integer host ID (without padding) from a given hostname.''' + try: + short_hostname = host.split('.')[0] + return int(re.findall("\\d+$", short_hostname)[-1]) + except IndexError: + raise ValueError("Host name must end in an integer") + + # Keeps already specified hosts, + # and adds or removes the hosts provided as an argument + def build_hostnames(self, changed_hosts, loadPreviousConfig=False): + existing_hosts = OrderedDict() + highest_host_id = 0 + # Load already existing hosts from the YAML + if loadPreviousConfig: + try: + for host in self.yaml_config['all']['hosts']: + # Read configuration of an existing host + hostConfig = self.yaml_config['all']['hosts'][host] + existing_hosts[host] = hostConfig + # If the existing host seems + # to have been created automatically, detect its ID + if host.startswith(HOST_PREFIX): + host_id = self.get_host_id(host) + if host_id > highest_host_id: + highest_host_id = host_id + except Exception as e: + # I am assuming we are catching automatically + # created hosts without IDs + print(e) + sys.exit(1) + + # FIXME(mattymo): Fix condition where delete then add reuses highest id + next_host_id = highest_host_id + 1 + next_host = "" + + all_hosts = existing_hosts.copy() + for host in changed_hosts: + # Delete the host from config the hostname/IP has a "-" prefix + if host[0] == "-": + realhost = host[1:] + if self.exists_hostname(all_hosts, realhost): + self.debug("Marked {0} for deletion.".format(realhost)) + all_hosts.pop(realhost) + elif self.exists_ip(all_hosts, realhost): + self.debug("Marked {0} for deletion.".format(realhost)) + self.delete_host_by_ip(all_hosts, realhost) + # Host/Argument starts with a digit, + # then we assume its an IP address + elif host[0].isdigit(): + if ',' in host: + ip, access_ip = host.split(',') + else: + ip = host + access_ip = host + if self.exists_hostname(all_hosts, host): + self.debug("Skipping existing host {0}.".format(host)) + continue + elif self.exists_ip(all_hosts, ip): + self.debug("Skipping existing host {0}.".format(ip)) + continue + + if USE_REAL_HOSTNAME: + cmd = ("ssh -oStrictHostKeyChecking=no " + + access_ip + " 'hostname -s'") + next_host = subprocess.check_output(cmd, shell=True) + next_host = next_host.strip().decode('ascii') + else: + # Generates a hostname because we have only an IP address + next_host = "{0}{1}".format(HOST_PREFIX, next_host_id) + next_host_id += 1 + # Uses automatically generated node name + # in case we dont provide it. + all_hosts[next_host] = {'ansible_host': access_ip, + 'ip': ip, + 'access_ip': access_ip} + # Host/Argument starts with a letter, then we assume its a hostname + elif host[0].isalpha(): + if ',' in host: + try: + hostname, ip, access_ip = host.split(',') + except Exception: + hostname, ip = host.split(',') + access_ip = ip + if self.exists_hostname(all_hosts, host): + self.debug("Skipping existing host {0}.".format(host)) + continue + elif self.exists_ip(all_hosts, ip): + self.debug("Skipping existing host {0}.".format(ip)) + continue + all_hosts[hostname] = {'ansible_host': access_ip, + 'ip': ip, + 'access_ip': access_ip} + return all_hosts + + # Expand IP ranges into individual addresses + def range2ips(self, hosts): + reworked_hosts = [] + + def ips(start_address, end_address): + try: + # Python 3.x + start = int(ip_address(start_address)) + end = int(ip_address(end_address)) + except Exception: + # Python 2.7 + start = int(ip_address(str(start_address))) + end = int(ip_address(str(end_address))) + return [ip_address(ip).exploded for ip in range(start, end + 1)] + + for host in hosts: + if '-' in host and not (host.startswith('-') or host[0].isalpha()): + start, end = host.strip().split('-') + try: + reworked_hosts.extend(ips(start, end)) + except ValueError: + raise Exception("Range of ip_addresses isn't valid") + else: + reworked_hosts.append(host) + return reworked_hosts + + def exists_hostname(self, existing_hosts, hostname): + return hostname in existing_hosts.keys() + + def exists_ip(self, existing_hosts, ip): + for host_opts in existing_hosts.values(): + if ip == self.get_ip_from_opts(host_opts): + return True + return False + + def delete_host_by_ip(self, existing_hosts, ip): + for hostname, host_opts in existing_hosts.items(): + if ip == self.get_ip_from_opts(host_opts): + del existing_hosts[hostname] + return + raise ValueError("Unable to find host by IP: {0}".format(ip)) + + def purge_invalid_hosts(self, hostnames, protected_names=[]): + for role in self.yaml_config['all']['children']: + if role != 'k8s_cluster' and self.yaml_config['all']['children'][role]['hosts']: # noqa + all_hosts = self.yaml_config['all']['children'][role]['hosts'].copy() # noqa + for host in all_hosts.keys(): + if host not in hostnames and host not in protected_names: + self.debug( + "Host {0} removed from role {1}".format(host, role)) # noqa + del self.yaml_config['all']['children'][role]['hosts'][host] # noqa + # purge from all + if self.yaml_config['all']['hosts']: + all_hosts = self.yaml_config['all']['hosts'].copy() + for host in all_hosts.keys(): + if host not in hostnames and host not in protected_names: + self.debug("Host {0} removed from role all".format(host)) + del self.yaml_config['all']['hosts'][host] + + def add_host_to_group(self, group, host, opts=""): + self.debug("adding host {0} to group {1}".format(host, group)) + if group == 'all': + if self.yaml_config['all']['hosts'] is None: + self.yaml_config['all']['hosts'] = {host: None} + self.yaml_config['all']['hosts'][host] = opts + elif group != 'k8s_cluster:children': + if self.yaml_config['all']['children'][group]['hosts'] is None: + self.yaml_config['all']['children'][group]['hosts'] = { + host: None} + else: + self.yaml_config['all']['children'][group]['hosts'][host] = None # noqa + + def set_kube_control_plane(self, hosts): + for host in hosts: + self.add_host_to_group('kube_control_plane', host) + + def set_all(self, hosts): + for host, opts in hosts.items(): + self.add_host_to_group('all', host, opts) + + def set_k8s_cluster(self): + k8s_cluster = {'children': {'kube_control_plane': None, + 'kube_node': None}} + self.yaml_config['all']['children']['k8s_cluster'] = k8s_cluster + + def set_calico_rr(self, hosts): + for host in hosts: + if host in self.yaml_config['all']['children']['kube_control_plane']: # noqa + self.debug("Not adding {0} to calico_rr group because it " + "conflicts with kube_control_plane " + "group".format(host)) + continue + if host in self.yaml_config['all']['children']['kube_node']: + self.debug("Not adding {0} to calico_rr group because it " + "conflicts with kube_node group".format(host)) + continue + self.add_host_to_group('calico_rr', host) + + def set_kube_node(self, hosts): + for host in hosts: + if len(self.yaml_config['all']['hosts']) >= SCALE_THRESHOLD: + if host in self.yaml_config['all']['children']['etcd']['hosts']: # noqa + self.debug("Not adding {0} to kube_node group because of " + "scale deployment and host is in etcd " + "group.".format(host)) + continue + if len(self.yaml_config['all']['hosts']) >= MASSIVE_SCALE_THRESHOLD: # noqa + if host in self.yaml_config['all']['children']['kube_control_plane']['hosts']: # noqa + self.debug("Not adding {0} to kube_node group because of " + "scale deployment and host is in " + "kube_control_plane group.".format(host)) + continue + self.add_host_to_group('kube_node', host) + + def set_etcd(self, hosts): + for host in hosts: + self.add_host_to_group('etcd', host) + + def load_file(self, files=None): + '''Directly loads JSON to inventory.''' + + if not files: + raise Exception("No input file specified.") + + import json + + for filename in list(files): + # Try JSON + try: + with open(filename, 'r') as f: + data = json.load(f) + except ValueError: + raise Exception("Cannot read %s as JSON, or CSV", filename) + + self.ensure_required_groups(ROLES) + self.set_k8s_cluster() + for group, hosts in data.items(): + self.ensure_required_groups([group]) + for host, opts in hosts.items(): + optstring = {'ansible_host': opts['ip'], + 'ip': opts['ip'], + 'access_ip': opts['ip']} + self.add_host_to_group('all', host, optstring) + self.add_host_to_group(group, host) + self.write_config(self.config_file) + + def parse_command(self, command, args=None): + if command == 'help': + self.show_help() + elif command == 'print_cfg': + self.print_config() + elif command == 'print_ips': + self.print_ips() + elif command == 'print_hostnames': + self.print_hostnames() + elif command == 'load': + self.load_file(args) + else: + raise Exception("Invalid command specified.") + + def show_help(self): + help_text = '''Usage: inventory.py ip1 [ip2 ...] +Examples: inventory.py 10.10.1.3 10.10.1.4 10.10.1.5 + +Available commands: +help - Display this message +print_cfg - Write inventory file to stdout +print_ips - Write a space-delimited list of IPs from "all" group +print_hostnames - Write a space-delimited list of Hostnames from "all" group +add - Adds specified hosts into an already existing inventory + +Advanced usage: +Create new or overwrite old inventory file: inventory.py 10.10.1.5 +Add another host after initial creation: inventory.py add 10.10.1.6 +Add range of hosts: inventory.py 10.10.1.3-10.10.1.5 +Add hosts with different ip and access ip: inventory.py 10.0.0.1,192.168.10.1 10.0.0.2,192.168.10.2 10.0.0.3,192.168.10.3 +Add hosts with a specific hostname, ip, and optional access ip: first,10.0.0.1,192.168.10.1 second,10.0.0.2 last,10.0.0.3 +Delete a host: inventory.py -10.10.1.3 +Delete a host by id: inventory.py -node1 + +Configurable env vars: +DEBUG Enable debug printing. Default: True +CONFIG_FILE File to write config to Default: ./inventory/sample/hosts.yaml +HOST_PREFIX Host prefix for generated hosts. Default: node +KUBE_CONTROL_HOSTS Set the number of kube-control-planes. Default: 2 +SCALE_THRESHOLD Separate ETCD role if # of nodes >= 50 +MASSIVE_SCALE_THRESHOLD Separate K8s control-plane and ETCD if # of nodes >= 200 +''' # noqa + print(help_text) + + def print_config(self): + yaml.dump(self.yaml_config, sys.stdout) + + def print_hostnames(self): + print(' '.join(self.yaml_config['all']['hosts'].keys())) + + def print_ips(self): + ips = [] + for host, opts in self.yaml_config['all']['hosts'].items(): + ips.append(self.get_ip_from_opts(opts)) + print(' '.join(ips)) + + +def main(argv=None): + if not argv: + argv = sys.argv[1:] + KubesprayInventory(argv, CONFIG_FILE) + return 0 + + +if __name__ == "__main__": + sys.exit(main()) diff --git a/kubespray/contrib/inventory_builder/requirements.txt b/kubespray/contrib/inventory_builder/requirements.txt new file mode 100644 index 0000000..c54501a --- /dev/null +++ b/kubespray/contrib/inventory_builder/requirements.txt @@ -0,0 +1,3 @@ +configparser>=3.3.0 +ipaddress +ruamel.yaml>=0.15.88 diff --git a/kubespray/contrib/inventory_builder/setup.cfg b/kubespray/contrib/inventory_builder/setup.cfg new file mode 100644 index 0000000..a775367 --- /dev/null +++ b/kubespray/contrib/inventory_builder/setup.cfg @@ -0,0 +1,3 @@ +[metadata] +name = kubespray-inventory-builder +version = 0.1 diff --git a/kubespray/contrib/inventory_builder/setup.py b/kubespray/contrib/inventory_builder/setup.py new file mode 100644 index 0000000..43c5ca1 --- /dev/null +++ b/kubespray/contrib/inventory_builder/setup.py @@ -0,0 +1,29 @@ +# Copyright (c) 2013 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or +# implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +# THIS FILE IS MANAGED BY THE GLOBAL REQUIREMENTS REPO - DO NOT EDIT +import setuptools + +# In python < 2.7.4, a lazy loading of package `pbr` will break +# setuptools if some other modules registered functions in `atexit`. +# solution from: http://bugs.python.org/issue15881#msg170215 +try: + import multiprocessing # noqa +except ImportError: + pass + +setuptools.setup( + setup_requires=[], + pbr=False) diff --git a/kubespray/contrib/inventory_builder/test-requirements.txt b/kubespray/contrib/inventory_builder/test-requirements.txt new file mode 100644 index 0000000..98a662a --- /dev/null +++ b/kubespray/contrib/inventory_builder/test-requirements.txt @@ -0,0 +1,3 @@ +hacking>=0.10.2 +mock>=1.3.0 +pytest>=2.8.0 diff --git a/kubespray/contrib/inventory_builder/tests/test_inventory.py b/kubespray/contrib/inventory_builder/tests/test_inventory.py new file mode 100644 index 0000000..5d6649d --- /dev/null +++ b/kubespray/contrib/inventory_builder/tests/test_inventory.py @@ -0,0 +1,595 @@ +# Copyright 2016 Mirantis, Inc. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +import inventory +from io import StringIO +import unittest +from unittest import mock + +from collections import OrderedDict +import sys + +path = "./contrib/inventory_builder/" +if path not in sys.path: + sys.path.append(path) + +import inventory # noqa + + +class TestInventoryPrintHostnames(unittest.TestCase): + + @mock.patch('ruamel.yaml.YAML.load') + def test_print_hostnames(self, load_mock): + mock_io = mock.mock_open(read_data='') + load_mock.return_value = OrderedDict({'all': {'hosts': { + 'node1': {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'}, + 'node2': {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'}}}}) + with mock.patch('builtins.open', mock_io): + with self.assertRaises(SystemExit) as cm: + with mock.patch('sys.stdout', new_callable=StringIO) as stdout: + inventory.KubesprayInventory( + changed_hosts=["print_hostnames"], + config_file="file") + self.assertEqual("node1 node2\n", stdout.getvalue()) + self.assertEqual(cm.exception.code, 0) + + +class TestInventory(unittest.TestCase): + @mock.patch('inventory.sys') + def setUp(self, sys_mock): + sys_mock.exit = mock.Mock() + super(TestInventory, self).setUp() + self.data = ['10.90.3.2', '10.90.3.3', '10.90.3.4'] + self.inv = inventory.KubesprayInventory() + + def test_get_ip_from_opts(self): + optstring = {'ansible_host': '10.90.3.2', + 'ip': '10.90.3.2', + 'access_ip': '10.90.3.2'} + expected = "10.90.3.2" + result = self.inv.get_ip_from_opts(optstring) + self.assertEqual(expected, result) + + def test_get_ip_from_opts_invalid(self): + optstring = "notanaddr=value something random!chars:D" + self.assertRaisesRegex(ValueError, "IP parameter not found", + self.inv.get_ip_from_opts, optstring) + + def test_ensure_required_groups(self): + groups = ['group1', 'group2'] + self.inv.ensure_required_groups(groups) + for group in groups: + self.assertIn(group, self.inv.yaml_config['all']['children']) + + def test_get_host_id(self): + hostnames = ['node99', 'no99de01', '01node01', 'node1.domain', + 'node3.xyz123.aaa'] + expected = [99, 1, 1, 1, 3] + for hostname, expected in zip(hostnames, expected): + result = self.inv.get_host_id(hostname) + self.assertEqual(expected, result) + + def test_get_host_id_invalid(self): + bad_hostnames = ['node', 'no99de', '01node', 'node.111111'] + for hostname in bad_hostnames: + self.assertRaisesRegex(ValueError, "Host name must end in an", + self.inv.get_host_id, hostname) + + def test_build_hostnames_add_duplicate(self): + changed_hosts = ['10.90.0.2'] + expected = OrderedDict([('node3', + {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'})]) + self.inv.yaml_config['all']['hosts'] = expected + result = self.inv.build_hostnames(changed_hosts, True) + self.assertEqual(expected, result) + + def test_build_hostnames_add_two(self): + changed_hosts = ['10.90.0.2', '10.90.0.3'] + expected = OrderedDict([ + ('node1', {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'}), + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'})]) + self.inv.yaml_config['all']['hosts'] = OrderedDict() + result = self.inv.build_hostnames(changed_hosts) + self.assertEqual(expected, result) + + def test_build_hostnames_add_three(self): + changed_hosts = ['10.90.0.2', '10.90.0.3', '10.90.0.4'] + expected = OrderedDict([ + ('node1', {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'}), + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'}), + ('node3', {'ansible_host': '10.90.0.4', + 'ip': '10.90.0.4', + 'access_ip': '10.90.0.4'})]) + result = self.inv.build_hostnames(changed_hosts) + self.assertEqual(expected, result) + + def test_build_hostnames_add_one(self): + changed_hosts = ['10.90.0.2'] + expected = OrderedDict([('node1', + {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'})]) + result = self.inv.build_hostnames(changed_hosts) + self.assertEqual(expected, result) + + def test_build_hostnames_delete_first(self): + changed_hosts = ['-10.90.0.2'] + existing_hosts = OrderedDict([ + ('node1', {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'}), + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'})]) + self.inv.yaml_config['all']['hosts'] = existing_hosts + expected = OrderedDict([ + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'})]) + result = self.inv.build_hostnames(changed_hosts, True) + self.assertEqual(expected, result) + + def test_build_hostnames_delete_by_hostname(self): + changed_hosts = ['-node1'] + existing_hosts = OrderedDict([ + ('node1', {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'}), + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'})]) + self.inv.yaml_config['all']['hosts'] = existing_hosts + expected = OrderedDict([ + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'})]) + result = self.inv.build_hostnames(changed_hosts, True) + self.assertEqual(expected, result) + + def test_exists_hostname_positive(self): + hostname = 'node1' + expected = True + existing_hosts = OrderedDict([ + ('node1', {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'}), + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'})]) + result = self.inv.exists_hostname(existing_hosts, hostname) + self.assertEqual(expected, result) + + def test_exists_hostname_negative(self): + hostname = 'node99' + expected = False + existing_hosts = OrderedDict([ + ('node1', {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'}), + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'})]) + result = self.inv.exists_hostname(existing_hosts, hostname) + self.assertEqual(expected, result) + + def test_exists_ip_positive(self): + ip = '10.90.0.2' + expected = True + existing_hosts = OrderedDict([ + ('node1', {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'}), + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'})]) + result = self.inv.exists_ip(existing_hosts, ip) + self.assertEqual(expected, result) + + def test_exists_ip_negative(self): + ip = '10.90.0.200' + expected = False + existing_hosts = OrderedDict([ + ('node1', {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'}), + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'})]) + result = self.inv.exists_ip(existing_hosts, ip) + self.assertEqual(expected, result) + + def test_delete_host_by_ip_positive(self): + ip = '10.90.0.2' + expected = OrderedDict([ + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'})]) + existing_hosts = OrderedDict([ + ('node1', {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'}), + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'})]) + self.inv.delete_host_by_ip(existing_hosts, ip) + self.assertEqual(expected, existing_hosts) + + def test_delete_host_by_ip_negative(self): + ip = '10.90.0.200' + existing_hosts = OrderedDict([ + ('node1', {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'}), + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'})]) + self.assertRaisesRegex(ValueError, "Unable to find host", + self.inv.delete_host_by_ip, existing_hosts, ip) + + def test_purge_invalid_hosts(self): + proper_hostnames = ['node1', 'node2'] + bad_host = 'doesnotbelong2' + existing_hosts = OrderedDict([ + ('node1', {'ansible_host': '10.90.0.2', + 'ip': '10.90.0.2', + 'access_ip': '10.90.0.2'}), + ('node2', {'ansible_host': '10.90.0.3', + 'ip': '10.90.0.3', + 'access_ip': '10.90.0.3'}), + ('doesnotbelong2', {'whateveropts=ilike'})]) + self.inv.yaml_config['all']['hosts'] = existing_hosts + self.inv.purge_invalid_hosts(proper_hostnames) + self.assertNotIn( + bad_host, self.inv.yaml_config['all']['hosts'].keys()) + + def test_add_host_to_group(self): + group = 'etcd' + host = 'node1' + opts = {'ip': '10.90.0.2'} + + self.inv.add_host_to_group(group, host, opts) + self.assertEqual( + self.inv.yaml_config['all']['children'][group]['hosts'].get(host), + None) + + def test_set_kube_control_plane(self): + group = 'kube_control_plane' + host = 'node1' + + self.inv.set_kube_control_plane([host]) + self.assertIn( + host, self.inv.yaml_config['all']['children'][group]['hosts']) + + def test_set_all(self): + hosts = OrderedDict([ + ('node1', 'opt1'), + ('node2', 'opt2')]) + + self.inv.set_all(hosts) + for host, opt in hosts.items(): + self.assertEqual( + self.inv.yaml_config['all']['hosts'].get(host), opt) + + def test_set_k8s_cluster(self): + group = 'k8s_cluster' + expected_hosts = ['kube_node', 'kube_control_plane'] + + self.inv.set_k8s_cluster() + for host in expected_hosts: + self.assertIn( + host, + self.inv.yaml_config['all']['children'][group]['children']) + + def test_set_kube_node(self): + group = 'kube_node' + host = 'node1' + + self.inv.set_kube_node([host]) + self.assertIn( + host, self.inv.yaml_config['all']['children'][group]['hosts']) + + def test_set_etcd(self): + group = 'etcd' + host = 'node1' + + self.inv.set_etcd([host]) + self.assertIn( + host, self.inv.yaml_config['all']['children'][group]['hosts']) + + def test_scale_scenario_one(self): + num_nodes = 50 + hosts = OrderedDict() + + for hostid in range(1, num_nodes+1): + hosts["node" + str(hostid)] = "" + + self.inv.set_all(hosts) + self.inv.set_etcd(list(hosts.keys())[0:3]) + self.inv.set_kube_control_plane(list(hosts.keys())[0:2]) + self.inv.set_kube_node(hosts.keys()) + for h in range(3): + self.assertFalse( + list(hosts.keys())[h] in + self.inv.yaml_config['all']['children']['kube_node']['hosts']) + + def test_scale_scenario_two(self): + num_nodes = 500 + hosts = OrderedDict() + + for hostid in range(1, num_nodes+1): + hosts["node" + str(hostid)] = "" + + self.inv.set_all(hosts) + self.inv.set_etcd(list(hosts.keys())[0:3]) + self.inv.set_kube_control_plane(list(hosts.keys())[3:5]) + self.inv.set_kube_node(hosts.keys()) + for h in range(5): + self.assertFalse( + list(hosts.keys())[h] in + self.inv.yaml_config['all']['children']['kube_node']['hosts']) + + def test_range2ips_range(self): + changed_hosts = ['10.90.0.2', '10.90.0.4-10.90.0.6', '10.90.0.8'] + expected = ['10.90.0.2', + '10.90.0.4', + '10.90.0.5', + '10.90.0.6', + '10.90.0.8'] + result = self.inv.range2ips(changed_hosts) + self.assertEqual(expected, result) + + def test_range2ips_incorrect_range(self): + host_range = ['10.90.0.4-a.9b.c.e'] + self.assertRaisesRegex(Exception, "Range of ip_addresses isn't valid", + self.inv.range2ips, host_range) + + def test_build_hostnames_create_with_one_different_ips(self): + changed_hosts = ['10.90.0.2,192.168.0.2'] + expected = OrderedDict([('node1', + {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'})]) + result = self.inv.build_hostnames(changed_hosts) + self.assertEqual(expected, result) + + def test_build_hostnames_create_with_two_different_ips(self): + changed_hosts = ['10.90.0.2,192.168.0.2', '10.90.0.3,192.168.0.3'] + expected = OrderedDict([ + ('node1', {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'}), + ('node2', {'ansible_host': '192.168.0.3', + 'ip': '10.90.0.3', + 'access_ip': '192.168.0.3'})]) + result = self.inv.build_hostnames(changed_hosts) + self.assertEqual(expected, result) + + def test_build_hostnames_create_with_three_different_ips(self): + changed_hosts = ['10.90.0.2,192.168.0.2', + '10.90.0.3,192.168.0.3', + '10.90.0.4,192.168.0.4'] + expected = OrderedDict([ + ('node1', {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'}), + ('node2', {'ansible_host': '192.168.0.3', + 'ip': '10.90.0.3', + 'access_ip': '192.168.0.3'}), + ('node3', {'ansible_host': '192.168.0.4', + 'ip': '10.90.0.4', + 'access_ip': '192.168.0.4'})]) + result = self.inv.build_hostnames(changed_hosts) + self.assertEqual(expected, result) + + def test_build_hostnames_overwrite_one_with_different_ips(self): + changed_hosts = ['10.90.0.2,192.168.0.2'] + expected = OrderedDict([('node1', + {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'})]) + existing = OrderedDict([('node5', + {'ansible_host': '192.168.0.5', + 'ip': '10.90.0.5', + 'access_ip': '192.168.0.5'})]) + self.inv.yaml_config['all']['hosts'] = existing + result = self.inv.build_hostnames(changed_hosts) + self.assertEqual(expected, result) + + def test_build_hostnames_overwrite_three_with_different_ips(self): + changed_hosts = ['10.90.0.2,192.168.0.2'] + expected = OrderedDict([('node1', + {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'})]) + existing = OrderedDict([ + ('node3', {'ansible_host': '192.168.0.3', + 'ip': '10.90.0.3', + 'access_ip': '192.168.0.3'}), + ('node4', {'ansible_host': '192.168.0.4', + 'ip': '10.90.0.4', + 'access_ip': '192.168.0.4'}), + ('node5', {'ansible_host': '192.168.0.5', + 'ip': '10.90.0.5', + 'access_ip': '192.168.0.5'})]) + self.inv.yaml_config['all']['hosts'] = existing + result = self.inv.build_hostnames(changed_hosts) + self.assertEqual(expected, result) + + def test_build_hostnames_different_ips_add_duplicate(self): + changed_hosts = ['10.90.0.2,192.168.0.2'] + expected = OrderedDict([('node3', + {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'})]) + existing = expected + self.inv.yaml_config['all']['hosts'] = existing + result = self.inv.build_hostnames(changed_hosts, True) + self.assertEqual(expected, result) + + def test_build_hostnames_add_two_different_ips_into_one_existing(self): + changed_hosts = ['10.90.0.3,192.168.0.3', '10.90.0.4,192.168.0.4'] + expected = OrderedDict([ + ('node2', {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'}), + ('node3', {'ansible_host': '192.168.0.3', + 'ip': '10.90.0.3', + 'access_ip': '192.168.0.3'}), + ('node4', {'ansible_host': '192.168.0.4', + 'ip': '10.90.0.4', + 'access_ip': '192.168.0.4'})]) + + existing = OrderedDict([ + ('node2', {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'})]) + self.inv.yaml_config['all']['hosts'] = existing + result = self.inv.build_hostnames(changed_hosts, True) + self.assertEqual(expected, result) + + def test_build_hostnames_add_two_different_ips_into_two_existing(self): + changed_hosts = ['10.90.0.4,192.168.0.4', '10.90.0.5,192.168.0.5'] + expected = OrderedDict([ + ('node2', {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'}), + ('node3', {'ansible_host': '192.168.0.3', + 'ip': '10.90.0.3', + 'access_ip': '192.168.0.3'}), + ('node4', {'ansible_host': '192.168.0.4', + 'ip': '10.90.0.4', + 'access_ip': '192.168.0.4'}), + ('node5', {'ansible_host': '192.168.0.5', + 'ip': '10.90.0.5', + 'access_ip': '192.168.0.5'})]) + + existing = OrderedDict([ + ('node2', {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'}), + ('node3', {'ansible_host': '192.168.0.3', + 'ip': '10.90.0.3', + 'access_ip': '192.168.0.3'})]) + self.inv.yaml_config['all']['hosts'] = existing + result = self.inv.build_hostnames(changed_hosts, True) + self.assertEqual(expected, result) + + def test_build_hostnames_add_two_different_ips_into_three_existing(self): + changed_hosts = ['10.90.0.5,192.168.0.5', '10.90.0.6,192.168.0.6'] + expected = OrderedDict([ + ('node2', {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'}), + ('node3', {'ansible_host': '192.168.0.3', + 'ip': '10.90.0.3', + 'access_ip': '192.168.0.3'}), + ('node4', {'ansible_host': '192.168.0.4', + 'ip': '10.90.0.4', + 'access_ip': '192.168.0.4'}), + ('node5', {'ansible_host': '192.168.0.5', + 'ip': '10.90.0.5', + 'access_ip': '192.168.0.5'}), + ('node6', {'ansible_host': '192.168.0.6', + 'ip': '10.90.0.6', + 'access_ip': '192.168.0.6'})]) + + existing = OrderedDict([ + ('node2', {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'}), + ('node3', {'ansible_host': '192.168.0.3', + 'ip': '10.90.0.3', + 'access_ip': '192.168.0.3'}), + ('node4', {'ansible_host': '192.168.0.4', + 'ip': '10.90.0.4', + 'access_ip': '192.168.0.4'})]) + self.inv.yaml_config['all']['hosts'] = existing + result = self.inv.build_hostnames(changed_hosts, True) + self.assertEqual(expected, result) + + # Add two IP addresses into a config that has + # three already defined IP addresses. One of the IP addresses + # is a duplicate. + def test_build_hostnames_add_two_duplicate_one_overlap(self): + changed_hosts = ['10.90.0.4,192.168.0.4', '10.90.0.5,192.168.0.5'] + expected = OrderedDict([ + ('node2', {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'}), + ('node3', {'ansible_host': '192.168.0.3', + 'ip': '10.90.0.3', + 'access_ip': '192.168.0.3'}), + ('node4', {'ansible_host': '192.168.0.4', + 'ip': '10.90.0.4', + 'access_ip': '192.168.0.4'}), + ('node5', {'ansible_host': '192.168.0.5', + 'ip': '10.90.0.5', + 'access_ip': '192.168.0.5'})]) + + existing = OrderedDict([ + ('node2', {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'}), + ('node3', {'ansible_host': '192.168.0.3', + 'ip': '10.90.0.3', + 'access_ip': '192.168.0.3'}), + ('node4', {'ansible_host': '192.168.0.4', + 'ip': '10.90.0.4', + 'access_ip': '192.168.0.4'})]) + self.inv.yaml_config['all']['hosts'] = existing + result = self.inv.build_hostnames(changed_hosts, True) + self.assertEqual(expected, result) + + # Add two duplicate IP addresses into a config that has + # three already defined IP addresses + def test_build_hostnames_add_two_duplicate_two_overlap(self): + changed_hosts = ['10.90.0.3,192.168.0.3', '10.90.0.4,192.168.0.4'] + expected = OrderedDict([ + ('node2', {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'}), + ('node3', {'ansible_host': '192.168.0.3', + 'ip': '10.90.0.3', + 'access_ip': '192.168.0.3'}), + ('node4', {'ansible_host': '192.168.0.4', + 'ip': '10.90.0.4', + 'access_ip': '192.168.0.4'})]) + + existing = OrderedDict([ + ('node2', {'ansible_host': '192.168.0.2', + 'ip': '10.90.0.2', + 'access_ip': '192.168.0.2'}), + ('node3', {'ansible_host': '192.168.0.3', + 'ip': '10.90.0.3', + 'access_ip': '192.168.0.3'}), + ('node4', {'ansible_host': '192.168.0.4', + 'ip': '10.90.0.4', + 'access_ip': '192.168.0.4'})]) + self.inv.yaml_config['all']['hosts'] = existing + result = self.inv.build_hostnames(changed_hosts, True) + self.assertEqual(expected, result) diff --git a/kubespray/contrib/inventory_builder/tox.ini b/kubespray/contrib/inventory_builder/tox.ini new file mode 100644 index 0000000..c9c7042 --- /dev/null +++ b/kubespray/contrib/inventory_builder/tox.ini @@ -0,0 +1,34 @@ +[tox] +minversion = 1.6 +skipsdist = True +envlist = pep8 + +[testenv] +allowlist_externals = py.test +usedevelop = True +deps = + -r{toxinidir}/requirements.txt + -r{toxinidir}/test-requirements.txt +setenv = VIRTUAL_ENV={envdir} +passenv = + http_proxy + HTTP_PROXY + https_proxy + HTTPS_PROXY + no_proxy + NO_PROXY +commands = pytest -vv #{posargs:./tests} + +[testenv:pep8] +usedevelop = False +allowlist_externals = bash +commands = + bash -c "find {toxinidir}/* -type f -name '*.py' -print0 | xargs -0 flake8" + +[testenv:venv] +commands = {posargs} + +[flake8] +show-source = true +builtins = _ +exclude=.venv,.git,.tox,dist,doc,*lib/python*,*egg diff --git a/kubespray/contrib/kvm-setup/README.md b/kubespray/contrib/kvm-setup/README.md new file mode 100644 index 0000000..559bc65 --- /dev/null +++ b/kubespray/contrib/kvm-setup/README.md @@ -0,0 +1,11 @@ +# Kubespray on KVM Virtual Machines hypervisor preparation + +A simple playbook to ensure your system has the right settings to enable Kubespray +deployment on VMs. + +This playbook does not create Virtual Machines, nor does it run Kubespray itself. + +## User creation + +If you want to create a user for running Kubespray deployment, you should specify +both `k8s_deployment_user` and `k8s_deployment_user_pkey_path`. diff --git a/kubespray/contrib/kvm-setup/group_vars/all b/kubespray/contrib/kvm-setup/group_vars/all new file mode 100644 index 0000000..d497c58 --- /dev/null +++ b/kubespray/contrib/kvm-setup/group_vars/all @@ -0,0 +1,2 @@ +#k8s_deployment_user: kubespray +#k8s_deployment_user_pkey_path: /tmp/ssh_rsa diff --git a/kubespray/contrib/kvm-setup/kvm-setup.yml b/kubespray/contrib/kvm-setup/kvm-setup.yml new file mode 100644 index 0000000..b8d4405 --- /dev/null +++ b/kubespray/contrib/kvm-setup/kvm-setup.yml @@ -0,0 +1,9 @@ +--- +- name: Prepare Hypervisor to later install kubespray VMs + hosts: localhost + gather_facts: False + become: yes + vars: + bootstrap_os: none + roles: + - { role: kvm-setup } diff --git a/kubespray/contrib/kvm-setup/roles/kvm-setup/tasks/main.yml b/kubespray/contrib/kvm-setup/roles/kvm-setup/tasks/main.yml new file mode 100644 index 0000000..3e8ade6 --- /dev/null +++ b/kubespray/contrib/kvm-setup/roles/kvm-setup/tasks/main.yml @@ -0,0 +1,30 @@ +--- + +- name: Install required packages + package: + name: "{{ item }}" + state: present + with_items: + - bind-utils + - ntp + when: ansible_os_family == "RedHat" + +- name: Install required packages + apt: + upgrade: yes + update_cache: yes + cache_valid_time: 3600 + name: "{{ item }}" + state: present + install_recommends: no + with_items: + - dnsutils + - ntp + when: ansible_os_family == "Debian" + +- name: Create deployment user if required + include_tasks: user.yml + when: k8s_deployment_user is defined + +- name: Set proper sysctl values + import_tasks: sysctl.yml diff --git a/kubespray/contrib/kvm-setup/roles/kvm-setup/tasks/sysctl.yml b/kubespray/contrib/kvm-setup/roles/kvm-setup/tasks/sysctl.yml new file mode 100644 index 0000000..52bc83f --- /dev/null +++ b/kubespray/contrib/kvm-setup/roles/kvm-setup/tasks/sysctl.yml @@ -0,0 +1,46 @@ +--- +- name: Load br_netfilter module + community.general.modprobe: + name: br_netfilter + state: present + register: br_netfilter + +- name: Add br_netfilter into /etc/modules + lineinfile: + dest: /etc/modules + state: present + line: 'br_netfilter' + when: br_netfilter is defined and ansible_os_family == 'Debian' + +- name: Add br_netfilter into /etc/modules-load.d/kubespray.conf + copy: + dest: /etc/modules-load.d/kubespray.conf + content: |- + ### This file is managed by Ansible + br-netfilter + owner: root + group: root + mode: 0644 + when: br_netfilter is defined + + +- name: Enable net.ipv4.ip_forward in sysctl + ansible.posix.sysctl: + name: net.ipv4.ip_forward + value: 1 + sysctl_file: "{{ sysctl_file_path }}" + state: present + reload: yes + +- name: Set bridge-nf-call-{arptables,iptables} to 0 + ansible.posix.sysctl: + name: "{{ item }}" + state: present + value: 0 + sysctl_file: "{{ sysctl_file_path }}" + reload: yes + with_items: + - net.bridge.bridge-nf-call-arptables + - net.bridge.bridge-nf-call-ip6tables + - net.bridge.bridge-nf-call-iptables + when: br_netfilter is defined diff --git a/kubespray/contrib/kvm-setup/roles/kvm-setup/tasks/user.yml b/kubespray/contrib/kvm-setup/roles/kvm-setup/tasks/user.yml new file mode 100644 index 0000000..c2d3123 --- /dev/null +++ b/kubespray/contrib/kvm-setup/roles/kvm-setup/tasks/user.yml @@ -0,0 +1,47 @@ +--- +- name: Create user {{ k8s_deployment_user }} + user: + name: "{{ k8s_deployment_user }}" + groups: adm + shell: /bin/bash + +- name: Ensure that .ssh exists + file: + path: "/home/{{ k8s_deployment_user }}/.ssh" + state: directory + owner: "{{ k8s_deployment_user }}" + group: "{{ k8s_deployment_user }}" + mode: 0700 + +- name: Configure sudo for deployment user + copy: + content: | + %{{ k8s_deployment_user }} ALL=(ALL) NOPASSWD: ALL + dest: "/etc/sudoers.d/55-k8s-deployment" + owner: root + group: root + mode: 0644 + +- name: Write private SSH key + copy: + src: "{{ k8s_deployment_user_pkey_path }}" + dest: "/home/{{ k8s_deployment_user }}/.ssh/id_rsa" + mode: 0400 + owner: "{{ k8s_deployment_user }}" + group: "{{ k8s_deployment_user }}" + when: k8s_deployment_user_pkey_path is defined + +- name: Write public SSH key + shell: "ssh-keygen -y -f /home/{{ k8s_deployment_user }}/.ssh/id_rsa \ + > /home/{{ k8s_deployment_user }}/.ssh/authorized_keys" + args: + creates: "/home/{{ k8s_deployment_user }}/.ssh/authorized_keys" + when: k8s_deployment_user_pkey_path is defined + +- name: Fix ssh-pub-key permissions + file: + path: "/home/{{ k8s_deployment_user }}/.ssh/authorized_keys" + mode: 0600 + owner: "{{ k8s_deployment_user }}" + group: "{{ k8s_deployment_user }}" + when: k8s_deployment_user_pkey_path is defined diff --git a/kubespray/contrib/misc/clusteradmin-rbac.yml b/kubespray/contrib/misc/clusteradmin-rbac.yml new file mode 100644 index 0000000..c02322f --- /dev/null +++ b/kubespray/contrib/misc/clusteradmin-rbac.yml @@ -0,0 +1,15 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: kubernetes-dashboard + labels: + k8s-app: kubernetes-dashboard +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cluster-admin +subjects: +- kind: ServiceAccount + name: kubernetes-dashboard + namespace: kube-system diff --git a/kubespray/contrib/mitogen/mitogen.yml b/kubespray/contrib/mitogen/mitogen.yml new file mode 100644 index 0000000..1ccc9a9 --- /dev/null +++ b/kubespray/contrib/mitogen/mitogen.yml @@ -0,0 +1,51 @@ +--- +- name: Check ansible version + import_playbook: kubernetes_sigs.kubespray.ansible_version + +- name: Install mitogen + hosts: localhost + strategy: linear + vars: + mitogen_version: 0.3.2 + mitogen_url: https://github.com/mitogen-hq/mitogen/archive/refs/tags/v{{ mitogen_version }}.tar.gz + ansible_connection: local + tasks: + - name: Create mitogen plugin dir + file: + path: "{{ item }}" + state: directory + mode: 0755 + become: false + loop: + - "{{ playbook_dir }}/plugins/mitogen" + - "{{ playbook_dir }}/dist" + + - name: Download mitogen release + get_url: + url: "{{ mitogen_url }}" + dest: "{{ playbook_dir }}/dist/mitogen_{{ mitogen_version }}.tar.gz" + validate_certs: true + mode: 0644 + + - name: Extract archive + unarchive: + src: "{{ playbook_dir }}/dist/mitogen_{{ mitogen_version }}.tar.gz" + dest: "{{ playbook_dir }}/dist/" + + - name: Copy plugin + ansible.posix.synchronize: + src: "{{ playbook_dir }}/dist/mitogen-{{ mitogen_version }}/" + dest: "{{ playbook_dir }}/plugins/mitogen" + + - name: Add strategy to ansible.cfg + community.general.ini_file: + path: ansible.cfg + mode: 0644 + section: "{{ item.section | d('defaults') }}" + option: "{{ item.option }}" + value: "{{ item.value }}" + with_items: + - option: strategy + value: mitogen_linear + - option: strategy_plugins + value: plugins/mitogen/ansible_mitogen/plugins/strategy diff --git a/kubespray/contrib/network-storage/glusterfs/README.md b/kubespray/contrib/network-storage/glusterfs/README.md new file mode 100644 index 0000000..bfe0a4d --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/README.md @@ -0,0 +1,92 @@ +# Deploying a Kubespray Kubernetes Cluster with GlusterFS + +You can either deploy using Ansible on its own by supplying your own inventory file or by using Terraform to create the VMs and then providing a dynamic inventory to Ansible. The following two sections are self-contained, you don't need to go through one to use the other. So, if you want to provision with Terraform, you can skip the **Using an Ansible inventory** section, and if you want to provision with a pre-built ansible inventory, you can neglect the **Using Terraform and Ansible** section. + +## Using an Ansible inventory + +In the same directory of this ReadMe file you should find a file named `inventory.example` which contains an example setup. Please note that, additionally to the Kubernetes nodes/masters, we define a set of machines for GlusterFS and we add them to the group `[gfs-cluster]`, which in turn is added to the larger `[network-storage]` group as a child group. + +Change that file to reflect your local setup (adding more machines or removing them and setting the adequate ip numbers), and save it to `inventory/sample/k8s_gfs_inventory`. Make sure that the settings on `inventory/sample/group_vars/all.yml` make sense with your deployment. Then execute change to the kubespray root folder, and execute (supposing that the machines are all using ubuntu): + +```shell +ansible-playbook -b --become-user=root -i inventory/sample/k8s_gfs_inventory --user=ubuntu ./cluster.yml +``` + +This will provision your Kubernetes cluster. Then, to provision and configure the GlusterFS cluster, from the same directory execute: + +```shell +ansible-playbook -b --become-user=root -i inventory/sample/k8s_gfs_inventory --user=ubuntu ./contrib/network-storage/glusterfs/glusterfs.yml +``` + +If your machines are not using Ubuntu, you need to change the `--user=ubuntu` to the correct user. Alternatively, if your Kubernetes machines are using one OS and your GlusterFS a different one, you can instead specify the `ansible_ssh_user=` variable in the inventory file that you just created, for each machine/VM: + +```shell +k8s-master-1 ansible_ssh_host=192.168.0.147 ip=192.168.0.147 ansible_ssh_user=core +k8s-master-node-1 ansible_ssh_host=192.168.0.148 ip=192.168.0.148 ansible_ssh_user=core +k8s-master-node-2 ansible_ssh_host=192.168.0.146 ip=192.168.0.146 ansible_ssh_user=core +``` + +## Using Terraform and Ansible + +First step is to fill in a `my-kubespray-gluster-cluster.tfvars` file with the specification desired for your cluster. An example with all required variables would look like: + +```ini +cluster_name = "cluster1" +number_of_k8s_masters = "1" +number_of_k8s_masters_no_floating_ip = "2" +number_of_k8s_nodes_no_floating_ip = "0" +number_of_k8s_nodes = "0" +public_key_path = "~/.ssh/my-desired-key.pub" +image = "Ubuntu 16.04" +ssh_user = "ubuntu" +flavor_k8s_node = "node-flavor-id-in-your-openstack" +flavor_k8s_master = "master-flavor-id-in-your-openstack" +network_name = "k8s-network" +floatingip_pool = "net_external" + +# GlusterFS variables +flavor_gfs_node = "gluster-flavor-id-in-your-openstack" +image_gfs = "Ubuntu 16.04" +number_of_gfs_nodes_no_floating_ip = "3" +gfs_volume_size_in_gb = "50" +ssh_user_gfs = "ubuntu" +``` + +As explained in the general terraform/openstack guide, you need to source your OpenStack credentials file, add your ssh-key to the ssh-agent and setup environment variables for terraform: + +```shell +$ source ~/.stackrc +$ eval $(ssh-agent -s) +$ ssh-add ~/.ssh/my-desired-key +$ echo Setting up Terraform creds && \ + export TF_VAR_username=${OS_USERNAME} && \ + export TF_VAR_password=${OS_PASSWORD} && \ + export TF_VAR_tenant=${OS_TENANT_NAME} && \ + export TF_VAR_auth_url=${OS_AUTH_URL} +``` + +Then, standing on the kubespray directory (root base of the Git checkout), issue the following terraform command to create the VMs for the cluster: + +```shell +terraform apply -state=contrib/terraform/openstack/terraform.tfstate -var-file=my-kubespray-gluster-cluster.tfvars contrib/terraform/openstack +``` + +This will create both your Kubernetes and Gluster VMs. Make sure that the ansible file `contrib/terraform/openstack/group_vars/all.yml` includes any ansible variable that you want to setup (like, for instance, the type of machine for bootstrapping). + +Then, provision your Kubernetes (kubespray) cluster with the following ansible call: + +```shell +ansible-playbook -b --become-user=root -i contrib/terraform/openstack/hosts ./cluster.yml +``` + +Finally, provision the glusterfs nodes and add the Persistent Volume setup for GlusterFS in Kubernetes through the following ansible call: + +```shell +ansible-playbook -b --become-user=root -i contrib/terraform/openstack/hosts ./contrib/network-storage/glusterfs/glusterfs.yml +``` + +If you need to destroy the cluster, you can run: + +```shell +terraform destroy -state=contrib/terraform/openstack/terraform.tfstate -var-file=my-kubespray-gluster-cluster.tfvars contrib/terraform/openstack +``` diff --git a/kubespray/contrib/network-storage/glusterfs/glusterfs.yml b/kubespray/contrib/network-storage/glusterfs/glusterfs.yml new file mode 100644 index 0000000..d5ade94 --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/glusterfs.yml @@ -0,0 +1,29 @@ +--- +- name: Bootstrap hosts + hosts: gfs-cluster + gather_facts: false + vars: + ansible_ssh_pipelining: false + roles: + - { role: bootstrap-os, tags: bootstrap-os} + +- name: Gather facts + hosts: all + gather_facts: true + +- name: Install glusterfs server + hosts: gfs-cluster + vars: + ansible_ssh_pipelining: true + roles: + - { role: glusterfs/server } + +- name: Install glusterfs servers + hosts: k8s_cluster + roles: + - { role: glusterfs/client } + +- name: Configure Kubernetes to use glusterfs + hosts: kube_control_plane[0] + roles: + - { role: kubernetes-pv } diff --git a/kubespray/contrib/network-storage/glusterfs/group_vars b/kubespray/contrib/network-storage/glusterfs/group_vars new file mode 120000 index 0000000..6a3f85e --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/group_vars @@ -0,0 +1 @@ +../../../inventory/local/group_vars \ No newline at end of file diff --git a/kubespray/contrib/network-storage/glusterfs/inventory.example b/kubespray/contrib/network-storage/glusterfs/inventory.example new file mode 100644 index 0000000..985647e --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/inventory.example @@ -0,0 +1,43 @@ +# ## Configure 'ip' variable to bind kubernetes services on a +# ## different ip than the default iface +# node1 ansible_ssh_host=95.54.0.12 # ip=10.3.0.1 +# node2 ansible_ssh_host=95.54.0.13 # ip=10.3.0.2 +# node3 ansible_ssh_host=95.54.0.14 # ip=10.3.0.3 +# node4 ansible_ssh_host=95.54.0.15 # ip=10.3.0.4 +# node5 ansible_ssh_host=95.54.0.16 # ip=10.3.0.5 +# node6 ansible_ssh_host=95.54.0.17 # ip=10.3.0.6 +# +# ## GlusterFS nodes +# ## Set disk_volume_device_1 to desired device for gluster brick, if different to /dev/vdb (default). +# ## As in the previous case, you can set ip to give direct communication on internal IPs +# gfs_node1 ansible_ssh_host=95.54.0.18 # disk_volume_device_1=/dev/vdc ip=10.3.0.7 +# gfs_node2 ansible_ssh_host=95.54.0.19 # disk_volume_device_1=/dev/vdc ip=10.3.0.8 +# gfs_node3 ansible_ssh_host=95.54.0.20 # disk_volume_device_1=/dev/vdc ip=10.3.0.9 + +# [kube_control_plane] +# node1 +# node2 + +# [etcd] +# node1 +# node2 +# node3 + +# [kube_node] +# node2 +# node3 +# node4 +# node5 +# node6 + +# [k8s_cluster:children] +# kube_node +# kube_control_plane + +# [gfs-cluster] +# gfs_node1 +# gfs_node2 +# gfs_node3 + +# [network-storage:children] +# gfs-cluster diff --git a/kubespray/contrib/network-storage/glusterfs/roles/bootstrap-os b/kubespray/contrib/network-storage/glusterfs/roles/bootstrap-os new file mode 120000 index 0000000..44dbbe4 --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/bootstrap-os @@ -0,0 +1 @@ +../../../../roles/bootstrap-os \ No newline at end of file diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/README.md b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/README.md new file mode 100644 index 0000000..dda243d --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/README.md @@ -0,0 +1,50 @@ +# Ansible Role: GlusterFS + +[![Build Status](https://travis-ci.org/geerlingguy/ansible-role-glusterfs.svg?branch=master)](https://travis-ci.org/geerlingguy/ansible-role-glusterfs) + +Installs and configures GlusterFS on Linux. + +## Requirements + +For GlusterFS to connect between servers, TCP ports `24007`, `24008`, and `24009`/`49152`+ (that port, plus an additional incremented port for each additional server in the cluster; the latter if GlusterFS is version 3.4+), and TCP/UDP port `111` must be open. You can open these using whatever firewall you wish (this can easily be configured using the `geerlingguy.firewall` role). + +This role performs basic installation and setup of Gluster, but it does not configure or mount bricks (volumes), since that step is easier to do in a series of plays in your own playbook. Ansible 1.9+ includes the [`gluster_volume`](https://docs.ansible.com/ansible/latest/collections/gluster/gluster/gluster_volume_module.html) module to ease the management of Gluster volumes. + +## Role Variables + +Available variables are listed below, along with default values (see `defaults/main.yml`): + +```yaml +glusterfs_default_release: "" +``` + +You can specify a `default_release` for apt on Debian/Ubuntu by overriding this variable. This is helpful if you need a different package or version for the main GlusterFS packages (e.g. GlusterFS 3.5.x instead of 3.2.x with the `wheezy-backports` default release on Debian Wheezy). + +```yaml +glusterfs_ppa_use: yes +glusterfs_ppa_version: "3.5" +``` + +For Ubuntu, specify whether to use the official Gluster PPA, and which version of the PPA to use. See Gluster's [Getting Started Guide](https://docs.gluster.org/en/latest/Quick-Start-Guide/Quickstart/) for more info. + +## Dependencies + +None. + +## Example Playbook + +```yaml + - hosts: server + roles: + - geerlingguy.glusterfs +``` + +For a real-world use example, read through [Simple GlusterFS Setup with Ansible](http://www.jeffgeerling.com/blog/simple-glusterfs-setup-ansible), a blog post by this role's author, which is included in Chapter 8 of [Ansible for DevOps](https://www.ansiblefordevops.com/). + +## License + +MIT / BSD + +## Author Information + +This role was created in 2015 by [Jeff Geerling](http://www.jeffgeerling.com/), author of [Ansible for DevOps](https://www.ansiblefordevops.com/). diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/defaults/main.yml b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/defaults/main.yml new file mode 100644 index 0000000..b9f0d2d --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/defaults/main.yml @@ -0,0 +1,11 @@ +--- +# For Ubuntu. +glusterfs_default_release: "" +glusterfs_ppa_use: yes +glusterfs_ppa_version: "4.1" + +# Gluster configuration. +gluster_mount_dir: /mnt/gluster +gluster_volume_node_mount_dir: /mnt/xfs-drive-gluster +gluster_brick_dir: "{{ gluster_volume_node_mount_dir }}/brick" +gluster_brick_name: gluster diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/meta/main.yml b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/meta/main.yml new file mode 100644 index 0000000..b7fe496 --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/meta/main.yml @@ -0,0 +1,30 @@ +--- +dependencies: [] + +galaxy_info: + author: geerlingguy + description: GlusterFS installation for Linux. + company: "Midwestern Mac, LLC" + license: "license (BSD, MIT)" + min_ansible_version: "2.0" + platforms: + - name: EL + versions: + - "6" + - "7" + - name: Ubuntu + versions: + - precise + - trusty + - xenial + - name: Debian + versions: + - wheezy + - jessie + galaxy_tags: + - system + - networking + - cloud + - clustering + - files + - sharing diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/tasks/main.yml b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/tasks/main.yml new file mode 100644 index 0000000..248f21e --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/tasks/main.yml @@ -0,0 +1,21 @@ +--- +# This is meant for Ubuntu and RedHat installations, where apparently the glusterfs-client is not used from inside +# hyperkube and needs to be installed as part of the system. + +# Setup/install tasks. +- name: Setup RedHat distros for glusterfs + include_tasks: setup-RedHat.yml + when: ansible_os_family == 'RedHat' and groups['gfs-cluster'] is defined + +- name: Setup Debian distros for glusterfs + include_tasks: setup-Debian.yml + when: ansible_os_family == 'Debian' and groups['gfs-cluster'] is defined + +- name: Ensure Gluster mount directories exist. + file: + path: "{{ item }}" + state: directory + mode: 0775 + with_items: + - "{{ gluster_mount_dir }}" + when: ansible_os_family in ["Debian","RedHat"] and groups['gfs-cluster'] is defined diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/tasks/setup-Debian.yml b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/tasks/setup-Debian.yml new file mode 100644 index 0000000..da7a4d8 --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/tasks/setup-Debian.yml @@ -0,0 +1,24 @@ +--- +- name: Add PPA for GlusterFS. + apt_repository: + repo: 'ppa:gluster/glusterfs-{{ glusterfs_ppa_version }}' + state: present + update_cache: yes + register: glusterfs_ppa_added + when: glusterfs_ppa_use + +- name: Ensure GlusterFS client will reinstall if the PPA was just added. # noqa no-handler + apt: + name: "{{ item }}" + state: absent + with_items: + - glusterfs-client + when: glusterfs_ppa_added.changed + +- name: Ensure GlusterFS client is installed. + apt: + name: "{{ item }}" + state: present + default_release: "{{ glusterfs_default_release }}" + with_items: + - glusterfs-client diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/tasks/setup-RedHat.yml b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/tasks/setup-RedHat.yml new file mode 100644 index 0000000..d2ee36a --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/client/tasks/setup-RedHat.yml @@ -0,0 +1,14 @@ +--- +- name: Install Prerequisites + package: + name: "{{ item }}" + state: present + with_items: + - "centos-release-gluster{{ glusterfs_default_release }}" + +- name: Install Packages + package: + name: "{{ item }}" + state: present + with_items: + - glusterfs-client diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/defaults/main.yml b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/defaults/main.yml new file mode 100644 index 0000000..ef9a71e --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/defaults/main.yml @@ -0,0 +1,13 @@ +--- +# For Ubuntu. +glusterfs_default_release: "" +glusterfs_ppa_use: yes +glusterfs_ppa_version: "3.12" + +# Gluster configuration. +gluster_mount_dir: /mnt/gluster +gluster_volume_node_mount_dir: /mnt/xfs-drive-gluster +gluster_brick_dir: "{{ gluster_volume_node_mount_dir }}/brick" +gluster_brick_name: gluster +# Default device to mount for xfs formatting, terraform overrides this by setting the variable in the inventory. +disk_volume_device_1: /dev/vdb diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/meta/main.yml b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/meta/main.yml new file mode 100644 index 0000000..b7fe496 --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/meta/main.yml @@ -0,0 +1,30 @@ +--- +dependencies: [] + +galaxy_info: + author: geerlingguy + description: GlusterFS installation for Linux. + company: "Midwestern Mac, LLC" + license: "license (BSD, MIT)" + min_ansible_version: "2.0" + platforms: + - name: EL + versions: + - "6" + - "7" + - name: Ubuntu + versions: + - precise + - trusty + - xenial + - name: Debian + versions: + - wheezy + - jessie + galaxy_tags: + - system + - networking + - cloud + - clustering + - files + - sharing diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/tasks/main.yml b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/tasks/main.yml new file mode 100644 index 0000000..50f849c --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/tasks/main.yml @@ -0,0 +1,113 @@ +--- +# Include variables and define needed variables. +- name: Include OS-specific variables. + include_vars: "{{ ansible_os_family }}.yml" + +# Install xfs package +- name: Install xfs Debian + apt: + name: xfsprogs + state: present + when: ansible_os_family == "Debian" + +- name: Install xfs RedHat + package: + name: xfsprogs + state: present + when: ansible_os_family == "RedHat" + +# Format external volumes in xfs +- name: Format volumes in xfs + community.general.filesystem: + fstype: xfs + dev: "{{ disk_volume_device_1 }}" + +# Mount external volumes +- name: Mounting new xfs filesystem + ansible.posix.mount: + name: "{{ gluster_volume_node_mount_dir }}" + src: "{{ disk_volume_device_1 }}" + fstype: xfs + state: mounted + +# Setup/install tasks. +- name: Setup RedHat distros for glusterfs + include_tasks: setup-RedHat.yml + when: ansible_os_family == 'RedHat' + +- name: Setup Debian distros for glusterfs + include_tasks: setup-Debian.yml + when: ansible_os_family == 'Debian' + +- name: Ensure GlusterFS is started and enabled at boot. + service: + name: "{{ glusterfs_daemon }}" + state: started + enabled: yes + +- name: Ensure Gluster brick and mount directories exist. + file: + path: "{{ item }}" + state: directory + mode: 0775 + with_items: + - "{{ gluster_brick_dir }}" + - "{{ gluster_mount_dir }}" + +- name: Configure Gluster volume with replicas + gluster.gluster.gluster_volume: + state: present + name: "{{ gluster_brick_name }}" + brick: "{{ gluster_brick_dir }}" + replicas: "{{ groups['gfs-cluster'] | length }}" + cluster: "{% for item in groups['gfs-cluster'] -%}{{ hostvars[item]['ip'] | default(hostvars[item].ansible_default_ipv4['address']) }}{% if not loop.last %},{% endif %}{%- endfor %}" + host: "{{ inventory_hostname }}" + force: yes + run_once: true + when: groups['gfs-cluster'] | length > 1 + +- name: Configure Gluster volume without replicas + gluster.gluster.gluster_volume: + state: present + name: "{{ gluster_brick_name }}" + brick: "{{ gluster_brick_dir }}" + cluster: "{% for item in groups['gfs-cluster'] -%}{{ hostvars[item]['ip'] | default(hostvars[item].ansible_default_ipv4['address']) }}{% if not loop.last %},{% endif %}{%- endfor %}" + host: "{{ inventory_hostname }}" + force: yes + run_once: true + when: groups['gfs-cluster'] | length <= 1 + +- name: Mount glusterfs to retrieve disk size + ansible.posix.mount: + name: "{{ gluster_mount_dir }}" + src: "{{ ip | default(ansible_default_ipv4['address']) }}:/gluster" + fstype: glusterfs + opts: "defaults,_netdev" + state: mounted + when: groups['gfs-cluster'] is defined and inventory_hostname == groups['gfs-cluster'][0] + +- name: Get Gluster disk size + setup: + filter: ansible_mounts + register: mounts_data + when: groups['gfs-cluster'] is defined and inventory_hostname == groups['gfs-cluster'][0] + +- name: Set Gluster disk size to variable + set_fact: + gluster_disk_size_gb: "{{ (mounts_data.ansible_facts.ansible_mounts | selectattr('mount', 'equalto', gluster_mount_dir) | map(attribute='size_total') | first | int / (1024 * 1024 * 1024)) | int }}" + when: groups['gfs-cluster'] is defined and inventory_hostname == groups['gfs-cluster'][0] + +- name: Create file on GlusterFS + template: + dest: "{{ gluster_mount_dir }}/.test-file.txt" + src: test-file.txt + mode: 0644 + when: groups['gfs-cluster'] is defined and inventory_hostname == groups['gfs-cluster'][0] + +- name: Unmount glusterfs + ansible.posix.mount: + name: "{{ gluster_mount_dir }}" + fstype: glusterfs + src: "{{ ip | default(ansible_default_ipv4['address']) }}:/gluster" + state: unmounted + when: groups['gfs-cluster'] is defined and inventory_hostname == groups['gfs-cluster'][0] diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/tasks/setup-Debian.yml b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/tasks/setup-Debian.yml new file mode 100644 index 0000000..1047359 --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/tasks/setup-Debian.yml @@ -0,0 +1,26 @@ +--- +- name: Add PPA for GlusterFS. + apt_repository: + repo: 'ppa:gluster/glusterfs-{{ glusterfs_ppa_version }}' + state: present + update_cache: yes + register: glusterfs_ppa_added + when: glusterfs_ppa_use + +- name: Ensure GlusterFS will reinstall if the PPA was just added. # noqa no-handler + apt: + name: "{{ item }}" + state: absent + with_items: + - glusterfs-server + - glusterfs-client + when: glusterfs_ppa_added.changed + +- name: Ensure GlusterFS is installed. + apt: + name: "{{ item }}" + state: present + default_release: "{{ glusterfs_default_release }}" + with_items: + - glusterfs-server + - glusterfs-client diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/tasks/setup-RedHat.yml b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/tasks/setup-RedHat.yml new file mode 100644 index 0000000..5a4e09e --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/tasks/setup-RedHat.yml @@ -0,0 +1,15 @@ +--- +- name: Install Prerequisites + package: + name: "{{ item }}" + state: present + with_items: + - "centos-release-gluster{{ glusterfs_default_release }}" + +- name: Install Packages + package: + name: "{{ item }}" + state: present + with_items: + - glusterfs-server + - glusterfs-client diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/templates/test-file.txt b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/templates/test-file.txt new file mode 100644 index 0000000..16b14f5 --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/templates/test-file.txt @@ -0,0 +1 @@ +test file diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/vars/Debian.yml b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/vars/Debian.yml new file mode 100644 index 0000000..e931068 --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/vars/Debian.yml @@ -0,0 +1,2 @@ +--- +glusterfs_daemon: glusterd diff --git a/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/vars/RedHat.yml b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/vars/RedHat.yml new file mode 100644 index 0000000..e931068 --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/glusterfs/server/vars/RedHat.yml @@ -0,0 +1,2 @@ +--- +glusterfs_daemon: glusterd diff --git a/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/tasks/main.yaml b/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/tasks/main.yaml new file mode 100644 index 0000000..ed62e28 --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/tasks/main.yaml @@ -0,0 +1,23 @@ +--- +- name: Kubernetes Apps | Lay Down k8s GlusterFS Endpoint and PV + template: + src: "{{ item.file }}" + dest: "{{ kube_config_dir }}/{{ item.dest }}" + mode: 0644 + with_items: + - { file: glusterfs-kubernetes-endpoint.json.j2, type: ep, dest: glusterfs-kubernetes-endpoint.json} + - { file: glusterfs-kubernetes-pv.yml.j2, type: pv, dest: glusterfs-kubernetes-pv.yml} + - { file: glusterfs-kubernetes-endpoint-svc.json.j2, type: svc, dest: glusterfs-kubernetes-endpoint-svc.json} + register: gluster_pv + when: inventory_hostname == groups['kube_control_plane'][0] and groups['gfs-cluster'] is defined and hostvars[groups['gfs-cluster'][0]].gluster_disk_size_gb is defined + +- name: Kubernetes Apps | Set GlusterFS endpoint and PV + kube: + name: glusterfs + namespace: default + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/{{ item.item.dest }}" + state: "{{ item.changed | ternary('latest', 'present') }}" + with_items: "{{ gluster_pv.results }}" + when: inventory_hostname == groups['kube_control_plane'][0] and groups['gfs-cluster'] is defined diff --git a/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/templates/glusterfs-kubernetes-endpoint-svc.json.j2 b/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/templates/glusterfs-kubernetes-endpoint-svc.json.j2 new file mode 100644 index 0000000..3cb5118 --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/templates/glusterfs-kubernetes-endpoint-svc.json.j2 @@ -0,0 +1,12 @@ +{ + "kind": "Service", + "apiVersion": "v1", + "metadata": { + "name": "glusterfs" + }, + "spec": { + "ports": [ + {"port": 1} + ] + } +} diff --git a/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/templates/glusterfs-kubernetes-endpoint.json.j2 b/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/templates/glusterfs-kubernetes-endpoint.json.j2 new file mode 100644 index 0000000..36cc1cc --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/templates/glusterfs-kubernetes-endpoint.json.j2 @@ -0,0 +1,23 @@ +{ + "kind": "Endpoints", + "apiVersion": "v1", + "metadata": { + "name": "glusterfs" + }, + "subsets": [ + {% for host in groups['gfs-cluster'] %} + { + "addresses": [ + { + "ip": "{{hostvars[host]['ip']|default(hostvars[host].ansible_default_ipv4['address'])}}" + } + ], + "ports": [ + { + "port": 1 + } + ] + }{%- if not loop.last %}, {% endif -%} + {% endfor %} + ] +} diff --git a/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/templates/glusterfs-kubernetes-pv.yml.j2 b/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/templates/glusterfs-kubernetes-pv.yml.j2 new file mode 100644 index 0000000..f6ba435 --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/ansible/templates/glusterfs-kubernetes-pv.yml.j2 @@ -0,0 +1,14 @@ +apiVersion: v1 +kind: PersistentVolume +metadata: + name: glusterfs +spec: + capacity: + storage: "{{ hostvars[groups['gfs-cluster'][0]].gluster_disk_size_gb }}Gi" + accessModes: + - ReadWriteMany + glusterfs: + endpoints: glusterfs + path: gluster + readOnly: false + persistentVolumeReclaimPolicy: Retain diff --git a/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/meta/main.yaml b/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/meta/main.yaml new file mode 100644 index 0000000..a4ab33f --- /dev/null +++ b/kubespray/contrib/network-storage/glusterfs/roles/kubernetes-pv/meta/main.yaml @@ -0,0 +1,3 @@ +--- +dependencies: + - {role: kubernetes-pv/ansible, tags: apps} diff --git a/kubespray/contrib/network-storage/heketi/README.md b/kubespray/contrib/network-storage/heketi/README.md new file mode 100644 index 0000000..d5491d3 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/README.md @@ -0,0 +1,27 @@ +# Deploy Heketi/Glusterfs into Kubespray/Kubernetes + +This playbook aims to automate [this](https://github.com/heketi/heketi/blob/master/docs/admin/install-kubernetes.md) tutorial. It deploys heketi/glusterfs into kubernetes and sets up a storageclass. + +## Important notice + +> Due to resource limits on the current project maintainers and general lack of contributions we are considering placing Heketi into a [near-maintenance mode](https://github.com/heketi/heketi#important-notice) + +## Client Setup + +Heketi provides a CLI that provides users with a means to administer the deployment and configuration of GlusterFS in Kubernetes. [Download and install the heketi-cli](https://github.com/heketi/heketi/releases) on your client machine. + +## Install + +Copy the inventory.yml.sample over to inventory/sample/k8s_heketi_inventory.yml and change it according to your setup. + +```shell +ansible-playbook --ask-become -i inventory/sample/k8s_heketi_inventory.yml contrib/network-storage/heketi/heketi.yml +``` + +## Tear down + +```shell +ansible-playbook --ask-become -i inventory/sample/k8s_heketi_inventory.yml contrib/network-storage/heketi/heketi-tear-down.yml +``` + +Add `--extra-vars "heketi_remove_lvm=true"` to the command above to remove LVM packages from the system diff --git a/kubespray/contrib/network-storage/heketi/heketi-tear-down.yml b/kubespray/contrib/network-storage/heketi/heketi-tear-down.yml new file mode 100644 index 0000000..e64f085 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/heketi-tear-down.yml @@ -0,0 +1,11 @@ +--- +- name: Tear down heketi + hosts: kube_control_plane[0] + roles: + - { role: tear-down } + +- name: Teardown disks in heketi + hosts: heketi-node + become: yes + roles: + - { role: tear-down-disks } diff --git a/kubespray/contrib/network-storage/heketi/heketi.yml b/kubespray/contrib/network-storage/heketi/heketi.yml new file mode 100644 index 0000000..bc0c4d0 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/heketi.yml @@ -0,0 +1,12 @@ +--- +- name: Prepare heketi install + hosts: heketi-node + roles: + - { role: prepare } + +- name: Provision heketi + hosts: kube_control_plane[0] + tags: + - "provision" + roles: + - { role: provision } diff --git a/kubespray/contrib/network-storage/heketi/inventory.yml.sample b/kubespray/contrib/network-storage/heketi/inventory.yml.sample new file mode 100644 index 0000000..467788a --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/inventory.yml.sample @@ -0,0 +1,33 @@ +all: + vars: + heketi_admin_key: "11elfeinhundertundelf" + heketi_user_key: "!!einseinseins" + glusterfs_daemonset: + readiness_probe: + timeout_seconds: 3 + initial_delay_seconds: 3 + liveness_probe: + timeout_seconds: 3 + initial_delay_seconds: 10 + children: + k8s_cluster: + vars: + kubelet_fail_swap_on: false + children: + kube_control_plane: + hosts: + node1: + etcd: + hosts: + node2: + kube_node: + hosts: &kube_nodes + node1: + node2: + node3: + node4: + heketi-node: + vars: + disk_volume_device_1: "/dev/vdb" + hosts: + <<: *kube_nodes diff --git a/kubespray/contrib/network-storage/heketi/requirements.txt b/kubespray/contrib/network-storage/heketi/requirements.txt new file mode 100644 index 0000000..45c1e03 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/requirements.txt @@ -0,0 +1 @@ +jmespath diff --git a/kubespray/contrib/network-storage/heketi/roles/prepare/tasks/main.yml b/kubespray/contrib/network-storage/heketi/roles/prepare/tasks/main.yml new file mode 100644 index 0000000..20012b1 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/prepare/tasks/main.yml @@ -0,0 +1,24 @@ +--- +- name: "Load lvm kernel modules" + become: true + with_items: + - "dm_snapshot" + - "dm_mirror" + - "dm_thin_pool" + community.general.modprobe: + name: "{{ item }}" + state: "present" + +- name: "Install glusterfs mount utils (RedHat)" + become: true + package: + name: "glusterfs-fuse" + state: "present" + when: "ansible_os_family == 'RedHat'" + +- name: "Install glusterfs mount utils (Debian)" + become: true + apt: + name: "glusterfs-client" + state: "present" + when: "ansible_os_family == 'Debian'" diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/defaults/main.yml b/kubespray/contrib/network-storage/heketi/roles/provision/defaults/main.yml new file mode 100644 index 0000000..ed97d53 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/defaults/main.yml @@ -0,0 +1 @@ +--- diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/handlers/main.yml b/kubespray/contrib/network-storage/heketi/roles/provision/handlers/main.yml new file mode 100644 index 0000000..4e768ad --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/handlers/main.yml @@ -0,0 +1,3 @@ +--- +- name: "Stop port forwarding" + command: "killall " diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap.yml new file mode 100644 index 0000000..7b43300 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap.yml @@ -0,0 +1,64 @@ +--- +# Bootstrap heketi +- name: "Get state of heketi service, deployment and pods." + register: "initial_heketi_state" + changed_when: false + command: "{{ bin_dir }}/kubectl get services,deployments,pods --selector=deploy-heketi --output=json" + +- name: "Bootstrap heketi." + when: + - "(initial_heketi_state.stdout | from_json | json_query(\"items[?kind=='Service']\")) | length == 0" + - "(initial_heketi_state.stdout | from_json | json_query(\"items[?kind=='Deployment']\")) | length == 0" + - "(initial_heketi_state.stdout | from_json | json_query(\"items[?kind=='Pod']\")) | length == 0" + include_tasks: "bootstrap/deploy.yml" + +# Prepare heketi topology +- name: "Get heketi initial pod state." + register: "initial_heketi_pod" + command: "{{ bin_dir }}/kubectl get pods --selector=deploy-heketi=pod,glusterfs=heketi-pod,name=deploy-heketi --output=json" + changed_when: false + +- name: "Ensure heketi bootstrap pod is up." + assert: + that: "(initial_heketi_pod.stdout | from_json | json_query('items[*]')) | length == 1" + +- name: Store the initial heketi pod name + set_fact: + initial_heketi_pod_name: "{{ initial_heketi_pod.stdout | from_json | json_query(\"items[*].metadata.name | [0]\") }}" + +- name: "Test heketi topology." + changed_when: false + register: "heketi_topology" + command: "{{ bin_dir }}/kubectl exec {{ initial_heketi_pod_name }} -- heketi-cli --user admin --secret {{ heketi_admin_key }} topology info --json" + +- name: "Load heketi topology." + when: "heketi_topology.stdout | from_json | json_query(\"clusters[*].nodes[*]\") | flatten | length == 0" + include_tasks: "bootstrap/topology.yml" + +# Provision heketi database volume +- name: "Prepare heketi volumes." + include_tasks: "bootstrap/volumes.yml" + +# Remove bootstrap heketi +- name: "Tear down bootstrap." + include_tasks: "bootstrap/tear-down.yml" + +# Prepare heketi storage +- name: "Test heketi storage." + command: "{{ bin_dir }}/kubectl get secrets,endpoints,services,jobs --output=json" + changed_when: false + register: "heketi_storage_state" + +# ensure endpoints actually exist before trying to move database data to it +- name: "Create heketi storage." + include_tasks: "bootstrap/storage.yml" + vars: + secret_query: "items[?metadata.name=='heketi-storage-secret' && kind=='Secret']" + endpoints_query: "items[?metadata.name=='heketi-storage-endpoints' && kind=='Endpoints']" + service_query: "items[?metadata.name=='heketi-storage-endpoints' && kind=='Service']" + job_query: "items[?metadata.name=='heketi-storage-copy-job' && kind=='Job']" + when: + - "heketi_storage_state.stdout | from_json | json_query(secret_query) | length == 0" + - "heketi_storage_state.stdout | from_json | json_query(endpoints_query) | length == 0" + - "heketi_storage_state.stdout | from_json | json_query(service_query) | length == 0" + - "heketi_storage_state.stdout | from_json | json_query(job_query) | length == 0" diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/deploy.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/deploy.yml new file mode 100644 index 0000000..866fe30 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/deploy.yml @@ -0,0 +1,27 @@ +--- +- name: "Kubernetes Apps | Lay Down Heketi Bootstrap" + become: true + template: + src: "heketi-bootstrap.json.j2" + dest: "{{ kube_config_dir }}/heketi-bootstrap.json" + mode: 0640 + register: "rendering" +- name: "Kubernetes Apps | Install and configure Heketi Bootstrap" + kube: + name: "GlusterFS" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/heketi-bootstrap.json" + state: "{{ rendering.changed | ternary('latest', 'present') }}" +- name: "Wait for heketi bootstrap to complete." + changed_when: false + register: "initial_heketi_state" + vars: + initial_heketi_state: { stdout: "{}" } + pods_query: "items[?kind=='Pod'].status.conditions | [0][?type=='Ready'].status | [0]" + deployments_query: "items[?kind=='Deployment'].status.conditions | [0][?type=='Available'].status | [0]" + command: "{{ bin_dir }}/kubectl get services,deployments,pods --selector=deploy-heketi --output=json" + until: + - "initial_heketi_state.stdout | from_json | json_query(pods_query) == 'True'" + - "initial_heketi_state.stdout | from_json | json_query(deployments_query) == 'True'" + retries: 60 + delay: 5 diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/storage.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/storage.yml new file mode 100644 index 0000000..650c12d --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/storage.yml @@ -0,0 +1,33 @@ +--- +- name: "Test heketi storage." + command: "{{ bin_dir }}/kubectl get secrets,endpoints,services,jobs --output=json" + changed_when: false + register: "heketi_storage_state" +- name: "Create heketi storage." + kube: + name: "GlusterFS" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/heketi-storage-bootstrap.json" + state: "present" + vars: + secret_query: "items[?metadata.name=='heketi-storage-secret' && kind=='Secret']" + endpoints_query: "items[?metadata.name=='heketi-storage-endpoints' && kind=='Endpoints']" + service_query: "items[?metadata.name=='heketi-storage-endpoints' && kind=='Service']" + job_query: "items[?metadata.name=='heketi-storage-copy-job' && kind=='Job']" + when: + - "heketi_storage_state.stdout | from_json | json_query(secret_query) | length == 0" + - "heketi_storage_state.stdout | from_json | json_query(endpoints_query) | length == 0" + - "heketi_storage_state.stdout | from_json | json_query(service_query) | length == 0" + - "heketi_storage_state.stdout | from_json | json_query(job_query) | length == 0" + register: "heketi_storage_result" +- name: "Get state of heketi database copy job." + command: "{{ bin_dir }}/kubectl get jobs --output=json" + changed_when: false + register: "heketi_storage_state" + vars: + heketi_storage_state: { stdout: "{}" } + job_query: "items[?metadata.name=='heketi-storage-copy-job' && kind=='Job' && status.succeeded==1]" + until: + - "heketi_storage_state.stdout | from_json | json_query(job_query) | length == 1" + retries: 60 + delay: 5 diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/tear-down.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/tear-down.yml new file mode 100644 index 0000000..ad48882 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/tear-down.yml @@ -0,0 +1,14 @@ +--- +- name: "Get existing Heketi deploy resources." + command: "{{ bin_dir }}/kubectl get all --selector=\"deploy-heketi\" -o=json" + register: "heketi_resources" + changed_when: false +- name: "Delete bootstrap Heketi." + command: "{{ bin_dir }}/kubectl delete all,service,jobs,deployment,secret --selector=\"deploy-heketi\"" + when: "heketi_resources.stdout | from_json | json_query('items[*]') | length > 0" +- name: "Ensure there is nothing left over." + command: "{{ bin_dir }}/kubectl get all,service,jobs,deployment,secret --selector=\"deploy-heketi\" -o=json" + register: "heketi_result" + until: "heketi_result.stdout | from_json | json_query('items[*]') | length == 0" + retries: 60 + delay: 5 diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/topology.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/topology.yml new file mode 100644 index 0000000..2f3efd4 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/topology.yml @@ -0,0 +1,27 @@ +--- +- name: "Get heketi topology." + changed_when: false + register: "heketi_topology" + command: "{{ bin_dir }}/kubectl exec {{ initial_heketi_pod_name }} -- heketi-cli --user admin --secret {{ heketi_admin_key }} topology info --json" +- name: "Render heketi topology template." + become: true + vars: { nodes: "{{ groups['heketi-node'] }}" } + register: "render" + template: + src: "topology.json.j2" + dest: "{{ kube_config_dir }}/topology.json" + mode: 0644 +- name: "Copy topology configuration into container." + changed_when: false + command: "{{ bin_dir }}/kubectl cp {{ kube_config_dir }}/topology.json {{ initial_heketi_pod_name }}:/tmp/topology.json" +- name: "Load heketi topology." # noqa no-handler + when: "render.changed" + command: "{{ bin_dir }}/kubectl exec {{ initial_heketi_pod_name }} -- heketi-cli --user admin --secret {{ heketi_admin_key }} topology load --json=/tmp/topology.json" + register: "load_heketi" +- name: "Get heketi topology." + changed_when: false + register: "heketi_topology" + command: "{{ bin_dir }}/kubectl exec {{ initial_heketi_pod_name }} -- heketi-cli --user admin --secret {{ heketi_admin_key }} topology info --json" + until: "heketi_topology.stdout | from_json | json_query(\"clusters[*].nodes[*].devices[?state=='online'].id\") | flatten | length == groups['heketi-node'] | length" + retries: 60 + delay: 5 diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/volumes.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/volumes.yml new file mode 100644 index 0000000..6d26dfc --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/bootstrap/volumes.yml @@ -0,0 +1,41 @@ +--- +- name: "Get heketi volume ids." + command: "{{ bin_dir }}/kubectl exec {{ initial_heketi_pod_name }} -- heketi-cli --user admin --secret {{ heketi_admin_key }} volume list --json" + changed_when: false + register: "heketi_volumes" +- name: "Get heketi volumes." + changed_when: false + command: "{{ bin_dir }}/kubectl exec {{ initial_heketi_pod_name }} -- heketi-cli --user admin --secret {{ heketi_admin_key }} volume info {{ volume_id }} --json" + with_items: "{{ heketi_volumes.stdout | from_json | json_query(\"volumes[*]\") }}" + loop_control: { loop_var: "volume_id" } + register: "volumes_information" +- name: "Test heketi database volume." + set_fact: { heketi_database_volume_exists: true } + with_items: "{{ volumes_information.results }}" + loop_control: { loop_var: "volume_information" } + vars: { volume: "{{ volume_information.stdout | from_json }}" } + when: "volume.name == 'heketidbstorage'" +- name: "Provision database volume." + command: "{{ bin_dir }}/kubectl exec {{ initial_heketi_pod_name }} -- heketi-cli --user admin --secret {{ heketi_admin_key }} setup-openshift-heketi-storage" + when: "heketi_database_volume_exists is undefined" +- name: "Copy configuration from pod." + become: true + command: "{{ bin_dir }}/kubectl cp {{ initial_heketi_pod_name }}:/heketi-storage.json {{ kube_config_dir }}/heketi-storage-bootstrap.json" +- name: "Get heketi volume ids." + command: "{{ bin_dir }}/kubectl exec {{ initial_heketi_pod_name }} -- heketi-cli --user admin --secret {{ heketi_admin_key }} volume list --json" + changed_when: false + register: "heketi_volumes" +- name: "Get heketi volumes." + changed_when: false + command: "{{ bin_dir }}/kubectl exec {{ initial_heketi_pod_name }} -- heketi-cli --user admin --secret {{ heketi_admin_key }} volume info {{ volume_id }} --json" + with_items: "{{ heketi_volumes.stdout | from_json | json_query(\"volumes[*]\") }}" + loop_control: { loop_var: "volume_id" } + register: "volumes_information" +- name: "Test heketi database volume." + set_fact: { heketi_database_volume_created: true } + with_items: "{{ volumes_information.results }}" + loop_control: { loop_var: "volume_information" } + vars: { volume: "{{ volume_information.stdout | from_json }}" } + when: "volume.name == 'heketidbstorage'" +- name: "Ensure heketi database volume exists." + assert: { that: "heketi_database_volume_created is defined", msg: "Heketi database volume does not exist." } diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/cleanup.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/cleanup.yml new file mode 100644 index 0000000..238f29b --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/cleanup.yml @@ -0,0 +1,4 @@ +--- +- name: "Clean up left over jobs." + command: "{{ bin_dir }}/kubectl delete jobs,pods --selector=\"deploy-heketi\"" + changed_when: false diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/glusterfs.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/glusterfs.yml new file mode 100644 index 0000000..973c668 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/glusterfs.yml @@ -0,0 +1,44 @@ +--- +- name: "Kubernetes Apps | Lay Down GlusterFS Daemonset" + template: + src: "glusterfs-daemonset.json.j2" + dest: "{{ kube_config_dir }}/glusterfs-daemonset.json" + mode: 0644 + become: true + register: "rendering" +- name: "Kubernetes Apps | Install and configure GlusterFS daemonset" + kube: + name: "GlusterFS" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/glusterfs-daemonset.json" + state: "{{ rendering.changed | ternary('latest', 'present') }}" +- name: "Kubernetes Apps | Label GlusterFS nodes" + include_tasks: "glusterfs/label.yml" + with_items: "{{ groups['heketi-node'] }}" + loop_control: + loop_var: "node" +- name: "Kubernetes Apps | Wait for daemonset to become available." + register: "daemonset_state" + command: "{{ bin_dir }}/kubectl get daemonset glusterfs --output=json --ignore-not-found=true" + changed_when: false + vars: + daemonset_state: { stdout: "{}" } + ready: "{{ daemonset_state.stdout | from_json | json_query(\"status.numberReady\") }}" + desired: "{{ daemonset_state.stdout | from_json | json_query(\"status.desiredNumberScheduled\") }}" + until: "ready | int >= 3" + retries: 60 + delay: 5 + +- name: "Kubernetes Apps | Lay Down Heketi Service Account" + template: + src: "heketi-service-account.json.j2" + dest: "{{ kube_config_dir }}/heketi-service-account.json" + mode: 0644 + become: true + register: "rendering" +- name: "Kubernetes Apps | Install and configure Heketi Service Account" + kube: + name: "GlusterFS" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/heketi-service-account.json" + state: "{{ rendering.changed | ternary('latest', 'present') }}" diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/glusterfs/label.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/glusterfs/label.yml new file mode 100644 index 0000000..4cefd47 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/glusterfs/label.yml @@ -0,0 +1,19 @@ +--- +- name: Get storage nodes + register: "label_present" + command: "{{ bin_dir }}/kubectl get node --selector=storagenode=glusterfs,kubernetes.io/hostname={{ node }} --ignore-not-found=true" + changed_when: false + +- name: "Assign storage label" + when: "label_present.stdout_lines | length == 0" + command: "{{ bin_dir }}/kubectl label node {{ node }} storagenode=glusterfs" + +- name: Get storage nodes again + register: "label_present" + command: "{{ bin_dir }}/kubectl get node --selector=storagenode=glusterfs,kubernetes.io/hostname={{ node }} --ignore-not-found=true" + changed_when: false + +- name: Ensure the label has been set + assert: + that: "label_present | length > 0" + msg: "Node {{ node }} has not been assigned with label storagenode=glusterfs." diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/heketi.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/heketi.yml new file mode 100644 index 0000000..a8549df --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/heketi.yml @@ -0,0 +1,34 @@ +--- +- name: "Kubernetes Apps | Lay Down Heketi" + become: true + template: + src: "heketi-deployment.json.j2" + dest: "{{ kube_config_dir }}/heketi-deployment.json" + mode: 0644 + register: "rendering" + +- name: "Kubernetes Apps | Install and configure Heketi" + kube: + name: "GlusterFS" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/heketi-deployment.json" + state: "{{ rendering.changed | ternary('latest', 'present') }}" + +- name: "Ensure heketi is up and running." + changed_when: false + register: "heketi_state" + vars: + heketi_state: + stdout: "{}" + pods_query: "items[?kind=='Pod'].status.conditions|[0][?type=='Ready'].status|[0]" + deployments_query: "items[?kind=='Deployment'].status.conditions|[0][?type=='Available'].status|[0]" + command: "{{ bin_dir }}/kubectl get deployments,pods --selector=glusterfs --output=json" + until: + - "heketi_state.stdout | from_json | json_query(pods_query) == 'True'" + - "heketi_state.stdout | from_json | json_query(deployments_query) == 'True'" + retries: 60 + delay: 5 + +- name: Set the Heketi pod name + set_fact: + heketi_pod_name: "{{ heketi_state.stdout | from_json | json_query(\"items[?kind=='Pod'].metadata.name|[0]\") }}" diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/main.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/main.yml new file mode 100644 index 0000000..1feb27d --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/main.yml @@ -0,0 +1,30 @@ +--- +- name: "Kubernetes Apps | GlusterFS" + include_tasks: "glusterfs.yml" + +- name: "Kubernetes Apps | Heketi Secrets" + include_tasks: "secret.yml" + +- name: "Kubernetes Apps | Test Heketi" + register: "heketi_service_state" + command: "{{ bin_dir }}/kubectl get service heketi-storage-endpoints -o=name --ignore-not-found=true" + changed_when: false + +- name: "Kubernetes Apps | Bootstrap Heketi" + when: "heketi_service_state.stdout == \"\"" + include_tasks: "bootstrap.yml" + +- name: "Kubernetes Apps | Heketi" + include_tasks: "heketi.yml" + +- name: "Kubernetes Apps | Heketi Topology" + include_tasks: "topology.yml" + +- name: "Kubernetes Apps | Heketi Storage" + include_tasks: "storage.yml" + +- name: "Kubernetes Apps | Storage Class" + include_tasks: "storageclass.yml" + +- name: "Clean up" + include_tasks: "cleanup.yml" diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/secret.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/secret.yml new file mode 100644 index 0000000..c455b6f --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/secret.yml @@ -0,0 +1,45 @@ +--- +- name: Get clusterrolebindings + register: "clusterrolebinding_state" + command: "{{ bin_dir }}/kubectl get clusterrolebinding heketi-gluster-admin -o=name --ignore-not-found=true" + changed_when: false + +- name: "Kubernetes Apps | Deploy cluster role binding." + when: "clusterrolebinding_state.stdout | length == 0" + command: "{{ bin_dir }}/kubectl create clusterrolebinding heketi-gluster-admin --clusterrole=edit --serviceaccount=default:heketi-service-account" + +- name: Get clusterrolebindings again + register: "clusterrolebinding_state" + command: "{{ bin_dir }}/kubectl get clusterrolebinding heketi-gluster-admin -o=name --ignore-not-found=true" + changed_when: false + +- name: Make sure that clusterrolebindings are present now + assert: + that: "clusterrolebinding_state.stdout | length > 0" + msg: "Cluster role binding is not present." + +- name: Get the heketi-config-secret secret + register: "secret_state" + command: "{{ bin_dir }}/kubectl get secret heketi-config-secret -o=name --ignore-not-found=true" + changed_when: false + +- name: "Render Heketi secret configuration." + become: true + template: + src: "heketi.json.j2" + dest: "{{ kube_config_dir }}/heketi.json" + mode: 0644 + +- name: "Deploy Heketi config secret" + when: "secret_state.stdout | length == 0" + command: "{{ bin_dir }}/kubectl create secret generic heketi-config-secret --from-file={{ kube_config_dir }}/heketi.json" + +- name: Get the heketi-config-secret secret again + register: "secret_state" + command: "{{ bin_dir }}/kubectl get secret heketi-config-secret -o=name --ignore-not-found=true" + changed_when: false + +- name: Make sure the heketi-config-secret secret exists now + assert: + that: "secret_state.stdout | length > 0" + msg: "Heketi config secret is not present." diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/storage.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/storage.yml new file mode 100644 index 0000000..055e179 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/storage.yml @@ -0,0 +1,15 @@ +--- +- name: "Kubernetes Apps | Lay Down Heketi Storage" + become: true + vars: { nodes: "{{ groups['heketi-node'] }}" } + template: + src: "heketi-storage.json.j2" + dest: "{{ kube_config_dir }}/heketi-storage.json" + mode: 0644 + register: "rendering" +- name: "Kubernetes Apps | Install and configure Heketi Storage" + kube: + name: "GlusterFS" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/heketi-storage.json" + state: "{{ rendering.changed | ternary('latest', 'present') }}" diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/storageclass.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/storageclass.yml new file mode 100644 index 0000000..bd4f666 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/storageclass.yml @@ -0,0 +1,26 @@ +--- +- name: "Test storage class." + command: "{{ bin_dir }}/kubectl get storageclass gluster --ignore-not-found=true --output=json" + register: "storageclass" + changed_when: false +- name: "Test heketi service." + command: "{{ bin_dir }}/kubectl get service heketi --ignore-not-found=true --output=json" + register: "heketi_service" + changed_when: false +- name: "Ensure heketi service is available." + assert: { that: "heketi_service.stdout != \"\"" } +- name: "Render storage class configuration." + become: true + vars: + endpoint_address: "{{ (heketi_service.stdout | from_json).spec.clusterIP }}" + template: + src: "storageclass.yml.j2" + dest: "{{ kube_config_dir }}/storageclass.yml" + mode: 0644 + register: "rendering" +- name: "Kubernetes Apps | Install and configure Storace Class" + kube: + name: "GlusterFS" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/storageclass.yml" + state: "{{ rendering.changed | ternary('latest', 'present') }}" diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/tasks/topology.yml b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/topology.yml new file mode 100644 index 0000000..aa66208 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/tasks/topology.yml @@ -0,0 +1,26 @@ +--- +- name: "Get heketi topology." + register: "heketi_topology" + changed_when: false + command: "{{ bin_dir }}/kubectl exec {{ heketi_pod_name }} -- heketi-cli --user admin --secret {{ heketi_admin_key }} topology info --json" +- name: "Render heketi topology template." + become: true + vars: { nodes: "{{ groups['heketi-node'] }}" } + register: "rendering" + template: + src: "topology.json.j2" + dest: "{{ kube_config_dir }}/topology.json" + mode: 0644 +- name: "Copy topology configuration into container." # noqa no-handler + when: "rendering.changed" + command: "{{ bin_dir }}/kubectl cp {{ kube_config_dir }}/topology.json {{ heketi_pod_name }}:/tmp/topology.json" +- name: "Load heketi topology." # noqa no-handler + when: "rendering.changed" + command: "{{ bin_dir }}/kubectl exec {{ heketi_pod_name }} -- heketi-cli --user admin --secret {{ heketi_admin_key }} topology load --json=/tmp/topology.json" +- name: "Get heketi topology." + register: "heketi_topology" + changed_when: false + command: "{{ bin_dir }}/kubectl exec {{ heketi_pod_name }} -- heketi-cli --user admin --secret {{ heketi_admin_key }} topology info --json" + until: "heketi_topology.stdout | from_json | json_query(\"clusters[*].nodes[*].devices[?state=='online'].id\") | flatten | length == groups['heketi-node'] | length" + retries: 60 + delay: 5 diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/templates/glusterfs-daemonset.json.j2 b/kubespray/contrib/network-storage/heketi/roles/provision/templates/glusterfs-daemonset.json.j2 new file mode 100644 index 0000000..a14b31c --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/templates/glusterfs-daemonset.json.j2 @@ -0,0 +1,149 @@ +{ + "kind": "DaemonSet", + "apiVersion": "apps/v1", + "metadata": { + "name": "glusterfs", + "labels": { + "glusterfs": "deployment" + }, + "annotations": { + "description": "GlusterFS Daemon Set", + "tags": "glusterfs" + } + }, + "spec": { + "selector": { + "matchLabels": { + "glusterfs-node": "daemonset" + } + }, + "template": { + "metadata": { + "name": "glusterfs", + "labels": { + "glusterfs-node": "daemonset" + } + }, + "spec": { + "nodeSelector": { + "storagenode" : "glusterfs" + }, + "hostNetwork": true, + "containers": [ + { + "image": "gluster/gluster-centos:gluster4u0_centos7", + "imagePullPolicy": "IfNotPresent", + "name": "glusterfs", + "volumeMounts": [ + { + "name": "glusterfs-heketi", + "mountPath": "/var/lib/heketi" + }, + { + "name": "glusterfs-run", + "mountPath": "/run" + }, + { + "name": "glusterfs-lvm", + "mountPath": "/run/lvm" + }, + { + "name": "glusterfs-etc", + "mountPath": "/etc/glusterfs" + }, + { + "name": "glusterfs-logs", + "mountPath": "/var/log/glusterfs" + }, + { + "name": "glusterfs-config", + "mountPath": "/var/lib/glusterd" + }, + { + "name": "glusterfs-dev", + "mountPath": "/dev" + }, + { + "name": "glusterfs-cgroup", + "mountPath": "/sys/fs/cgroup" + } + ], + "securityContext": { + "capabilities": {}, + "privileged": true + }, + "readinessProbe": { + "timeoutSeconds": {{ glusterfs_daemonset.readiness_probe.timeout_seconds }}, + "initialDelaySeconds": {{ glusterfs_daemonset.readiness_probe.initial_delay_seconds }}, + "exec": { + "command": [ + "/bin/bash", + "-c", + "systemctl status glusterd.service" + ] + } + }, + "livenessProbe": { + "timeoutSeconds": {{ glusterfs_daemonset.liveness_probe.timeout_seconds }}, + "initialDelaySeconds": {{ glusterfs_daemonset.liveness_probe.initial_delay_seconds }}, + "exec": { + "command": [ + "/bin/bash", + "-c", + "systemctl status glusterd.service" + ] + } + } + } + ], + "volumes": [ + { + "name": "glusterfs-heketi", + "hostPath": { + "path": "/var/lib/heketi" + } + }, + { + "name": "glusterfs-run" + }, + { + "name": "glusterfs-lvm", + "hostPath": { + "path": "/run/lvm" + } + }, + { + "name": "glusterfs-etc", + "hostPath": { + "path": "/etc/glusterfs" + } + }, + { + "name": "glusterfs-logs", + "hostPath": { + "path": "/var/log/glusterfs" + } + }, + { + "name": "glusterfs-config", + "hostPath": { + "path": "/var/lib/glusterd" + } + }, + { + "name": "glusterfs-dev", + "hostPath": { + "path": "/dev" + } + }, + { + "name": "glusterfs-cgroup", + "hostPath": { + "path": "/sys/fs/cgroup" + } + } + ] + } + } + } +} diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-bootstrap.json.j2 b/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-bootstrap.json.j2 new file mode 100644 index 0000000..7a932d0 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-bootstrap.json.j2 @@ -0,0 +1,138 @@ +{ + "kind": "List", + "apiVersion": "v1", + "items": [ + { + "kind": "Service", + "apiVersion": "v1", + "metadata": { + "name": "deploy-heketi", + "labels": { + "glusterfs": "heketi-service", + "deploy-heketi": "support" + }, + "annotations": { + "description": "Exposes Heketi Service" + } + }, + "spec": { + "selector": { + "name": "deploy-heketi" + }, + "ports": [ + { + "name": "deploy-heketi", + "port": 8080, + "targetPort": 8080 + } + ] + } + }, + { + "kind": "Deployment", + "apiVersion": "apps/v1", + "metadata": { + "name": "deploy-heketi", + "labels": { + "glusterfs": "heketi-deployment", + "deploy-heketi": "deployment" + }, + "annotations": { + "description": "Defines how to deploy Heketi" + } + }, + "spec": { + "selector": { + "matchLabels": { + "name": "deploy-heketi" + } + }, + "replicas": 1, + "template": { + "metadata": { + "name": "deploy-heketi", + "labels": { + "name": "deploy-heketi", + "glusterfs": "heketi-pod", + "deploy-heketi": "pod" + } + }, + "spec": { + "serviceAccountName": "heketi-service-account", + "containers": [ + { + "image": "heketi/heketi:9", + "imagePullPolicy": "Always", + "name": "deploy-heketi", + "env": [ + { + "name": "HEKETI_EXECUTOR", + "value": "kubernetes" + }, + { + "name": "HEKETI_DB_PATH", + "value": "/var/lib/heketi/heketi.db" + }, + { + "name": "HEKETI_FSTAB", + "value": "/var/lib/heketi/fstab" + }, + { + "name": "HEKETI_SNAPSHOT_LIMIT", + "value": "14" + }, + { + "name": "HEKETI_KUBE_GLUSTER_DAEMONSET", + "value": "y" + } + ], + "ports": [ + { + "containerPort": 8080 + } + ], + "volumeMounts": [ + { + "name": "db", + "mountPath": "/var/lib/heketi" + }, + { + "name": "config", + "mountPath": "/etc/heketi" + } + ], + "readinessProbe": { + "timeoutSeconds": 3, + "initialDelaySeconds": 3, + "httpGet": { + "path": "/hello", + "port": 8080 + } + }, + "livenessProbe": { + "timeoutSeconds": 3, + "initialDelaySeconds": 10, + "httpGet": { + "path": "/hello", + "port": 8080 + } + } + } + ], + "volumes": [ + { + "name": "db" + }, + { + "name": "config", + "secret": { + "secretName": "heketi-config-secret" + } + } + ] + } + } + } + } + ] +} diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-deployment.json.j2 b/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-deployment.json.j2 new file mode 100644 index 0000000..8e09ce8 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-deployment.json.j2 @@ -0,0 +1,164 @@ +{ + "kind": "List", + "apiVersion": "v1", + "items": [ + { + "kind": "Secret", + "apiVersion": "v1", + "metadata": { + "name": "heketi-db-backup", + "labels": { + "glusterfs": "heketi-db", + "heketi": "db" + } + }, + "data": { + }, + "type": "Opaque" + }, + { + "kind": "Service", + "apiVersion": "v1", + "metadata": { + "name": "heketi", + "labels": { + "glusterfs": "heketi-service", + "deploy-heketi": "support" + }, + "annotations": { + "description": "Exposes Heketi Service" + } + }, + "spec": { + "selector": { + "name": "heketi" + }, + "ports": [ + { + "name": "heketi", + "port": 8080, + "targetPort": 8080 + } + ] + } + }, + { + "kind": "Deployment", + "apiVersion": "apps/v1", + "metadata": { + "name": "heketi", + "labels": { + "glusterfs": "heketi-deployment" + }, + "annotations": { + "description": "Defines how to deploy Heketi" + } + }, + "spec": { + "selector": { + "matchLabels": { + "name": "heketi" + } + }, + "replicas": 1, + "template": { + "metadata": { + "name": "heketi", + "labels": { + "name": "heketi", + "glusterfs": "heketi-pod" + } + }, + "spec": { + "serviceAccountName": "heketi-service-account", + "containers": [ + { + "image": "heketi/heketi:9", + "imagePullPolicy": "Always", + "name": "heketi", + "env": [ + { + "name": "HEKETI_EXECUTOR", + "value": "kubernetes" + }, + { + "name": "HEKETI_DB_PATH", + "value": "/var/lib/heketi/heketi.db" + }, + { + "name": "HEKETI_FSTAB", + "value": "/var/lib/heketi/fstab" + }, + { + "name": "HEKETI_SNAPSHOT_LIMIT", + "value": "14" + }, + { + "name": "HEKETI_KUBE_GLUSTER_DAEMONSET", + "value": "y" + } + ], + "ports": [ + { + "containerPort": 8080 + } + ], + "volumeMounts": [ + { + "mountPath": "/backupdb", + "name": "heketi-db-secret" + }, + { + "name": "db", + "mountPath": "/var/lib/heketi" + }, + { + "name": "config", + "mountPath": "/etc/heketi" + } + ], + "readinessProbe": { + "timeoutSeconds": 3, + "initialDelaySeconds": 3, + "httpGet": { + "path": "/hello", + "port": 8080 + } + }, + "livenessProbe": { + "timeoutSeconds": 3, + "initialDelaySeconds": 10, + "httpGet": { + "path": "/hello", + "port": 8080 + } + } + } + ], + "volumes": [ + { + "name": "db", + "glusterfs": { + "endpoints": "heketi-storage-endpoints", + "path": "heketidbstorage" + } + }, + { + "name": "heketi-db-secret", + "secret": { + "secretName": "heketi-db-backup" + } + }, + { + "name": "config", + "secret": { + "secretName": "heketi-config-secret" + } + } + ] + } + } + } + } + ] +} diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-service-account.json.j2 b/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-service-account.json.j2 new file mode 100644 index 0000000..1dbcb9e --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-service-account.json.j2 @@ -0,0 +1,7 @@ +{ + "apiVersion": "v1", + "kind": "ServiceAccount", + "metadata": { + "name": "heketi-service-account" + } +} diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-storage.json.j2 b/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-storage.json.j2 new file mode 100644 index 0000000..e985d25 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi-storage.json.j2 @@ -0,0 +1,54 @@ +{ + "apiVersion": "v1", + "kind": "List", + "items": [ + { + "kind": "Endpoints", + "apiVersion": "v1", + "metadata": { + "name": "heketi-storage-endpoints", + "creationTimestamp": null + }, + "subsets": [ +{% set nodeblocks = [] %} +{% for node in nodes %} +{% set nodeblock %} + { + "addresses": [ + { + "ip": "{{ hostvars[node].ip }}" + } + ], + "ports": [ + { + "port": 1 + } + ] + } +{% endset %} +{% if nodeblocks.append(nodeblock) %}{% endif %} +{% endfor %} +{{ nodeblocks|join(',') }} + ] + }, + { + "kind": "Service", + "apiVersion": "v1", + "metadata": { + "name": "heketi-storage-endpoints", + "creationTimestamp": null + }, + "spec": { + "ports": [ + { + "port": 1, + "targetPort": 0 + } + ] + }, + "status": { + "loadBalancer": {} + } + } + ] +} diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi.json.j2 b/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi.json.j2 new file mode 100644 index 0000000..5861b68 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/templates/heketi.json.j2 @@ -0,0 +1,44 @@ +{ + "_port_comment": "Heketi Server Port Number", + "port": "8080", + + "_use_auth": "Enable JWT authorization. Please enable for deployment", + "use_auth": true, + + "_jwt": "Private keys for access", + "jwt": { + "_admin": "Admin has access to all APIs", + "admin": { + "key": "{{ heketi_admin_key }}" + }, + "_user": "User only has access to /volumes endpoint", + "user": { + "key": "{{ heketi_user_key }}" + } + }, + + "_glusterfs_comment": "GlusterFS Configuration", + "glusterfs": { + "_executor_comment": "Execute plugin. Possible choices: mock, kubernetes, ssh", + "executor": "kubernetes", + + "_db_comment": "Database file name", + "db": "/var/lib/heketi/heketi.db", + + "kubeexec": { + "rebalance_on_expansion": true + }, + + "sshexec": { + "rebalance_on_expansion": true, + "keyfile": "/etc/heketi/private_key", + "fstab": "/etc/fstab", + "port": "22", + "user": "root", + "sudo": false + } + }, + + "_backup_db_to_kube_secret": "Backup the heketi database to a Kubernetes secret when running in Kubernetes. Default is off.", + "backup_db_to_kube_secret": false +} diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/templates/storageclass.yml.j2 b/kubespray/contrib/network-storage/heketi/roles/provision/templates/storageclass.yml.j2 new file mode 100644 index 0000000..c2b64cf --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/templates/storageclass.yml.j2 @@ -0,0 +1,12 @@ +--- +apiVersion: storage.k8s.io/v1 +kind: StorageClass +metadata: + name: gluster + annotations: + storageclass.beta.kubernetes.io/is-default-class: "true" +provisioner: kubernetes.io/glusterfs +parameters: + resturl: "http://{{ endpoint_address }}:8080" + restuser: "admin" + restuserkey: "{{ heketi_admin_key }}" diff --git a/kubespray/contrib/network-storage/heketi/roles/provision/templates/topology.json.j2 b/kubespray/contrib/network-storage/heketi/roles/provision/templates/topology.json.j2 new file mode 100644 index 0000000..c19ce32 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/provision/templates/topology.json.j2 @@ -0,0 +1,34 @@ +{ + "clusters": [ + { + "nodes": [ +{% set nodeblocks = [] %} +{% for node in nodes %} +{% set nodeblock %} + { + "node": { + "hostnames": { + "manage": [ + "{{ node }}" + ], + "storage": [ + "{{ hostvars[node].ip }}" + ] + }, + "zone": 1 + }, + "devices": [ + { + "name": "{{ hostvars[node]['disk_volume_device_1'] }}", + "destroydata": false + } + ] + } +{% endset %} +{% if nodeblocks.append(nodeblock) %}{% endif %} +{% endfor %} +{{ nodeblocks|join(',') }} + ] + } + ] +} diff --git a/kubespray/contrib/network-storage/heketi/roles/tear-down-disks/defaults/main.yml b/kubespray/contrib/network-storage/heketi/roles/tear-down-disks/defaults/main.yml new file mode 100644 index 0000000..c07ba2d --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/tear-down-disks/defaults/main.yml @@ -0,0 +1,2 @@ +--- +heketi_remove_lvm: false diff --git a/kubespray/contrib/network-storage/heketi/roles/tear-down-disks/tasks/main.yml b/kubespray/contrib/network-storage/heketi/roles/tear-down-disks/tasks/main.yml new file mode 100644 index 0000000..f3ca033 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/tear-down-disks/tasks/main.yml @@ -0,0 +1,52 @@ +--- +- name: "Install lvm utils (RedHat)" + become: true + package: + name: "lvm2" + state: "present" + when: "ansible_os_family == 'RedHat'" + +- name: "Install lvm utils (Debian)" + become: true + apt: + name: "lvm2" + state: "present" + when: "ansible_os_family == 'Debian'" + +- name: "Get volume group information." + environment: + PATH: "{{ ansible_env.PATH }}:/sbin" # Make sure we can workaround RH / CentOS conservative path management + become: true + shell: "pvs {{ disk_volume_device_1 }} --option vg_name | tail -n+2" + register: "volume_groups" + ignore_errors: true # noqa ignore-errors + changed_when: false + +- name: "Remove volume groups." + environment: + PATH: "{{ ansible_env.PATH }}:/sbin" # Make sure we can workaround RH / CentOS conservative path management + become: true + command: "vgremove {{ volume_group }} --yes" + with_items: "{{ volume_groups.stdout_lines }}" + loop_control: { loop_var: "volume_group" } + +- name: "Remove physical volume from cluster disks." + environment: + PATH: "{{ ansible_env.PATH }}:/sbin" # Make sure we can workaround RH / CentOS conservative path management + become: true + command: "pvremove {{ disk_volume_device_1 }} --yes" + ignore_errors: true # noqa ignore-errors + +- name: "Remove lvm utils (RedHat)" + become: true + package: + name: "lvm2" + state: "absent" + when: "ansible_os_family == 'RedHat' and heketi_remove_lvm" + +- name: "Remove lvm utils (Debian)" + become: true + apt: + name: "lvm2" + state: "absent" + when: "ansible_os_family == 'Debian' and heketi_remove_lvm" diff --git a/kubespray/contrib/network-storage/heketi/roles/tear-down/tasks/main.yml b/kubespray/contrib/network-storage/heketi/roles/tear-down/tasks/main.yml new file mode 100644 index 0000000..5c271e7 --- /dev/null +++ b/kubespray/contrib/network-storage/heketi/roles/tear-down/tasks/main.yml @@ -0,0 +1,51 @@ +--- +- name: Remove storage class. + command: "{{ bin_dir }}/kubectl delete storageclass gluster" + ignore_errors: true # noqa ignore-errors +- name: Tear down heketi. + command: "{{ bin_dir }}/kubectl delete all,service,jobs,deployment,secret --selector=\"glusterfs=heketi-pod\"" + ignore_errors: true # noqa ignore-errors +- name: Tear down heketi. + command: "{{ bin_dir }}/kubectl delete all,service,jobs,deployment,secret --selector=\"glusterfs=heketi-deployment\"" + ignore_errors: true # noqa ignore-errors +- name: Tear down bootstrap. + include_tasks: "../../provision/tasks/bootstrap/tear-down.yml" +- name: Ensure there is nothing left over. + command: "{{ bin_dir }}/kubectl get all,service,jobs,deployment,secret --selector=\"glusterfs=heketi-pod\" -o=json" + register: "heketi_result" + until: "heketi_result.stdout | from_json | json_query('items[*]') | length == 0" + retries: 60 + delay: 5 +- name: Ensure there is nothing left over. + command: "{{ bin_dir }}/kubectl get all,service,jobs,deployment,secret --selector=\"glusterfs=heketi-deployment\" -o=json" + register: "heketi_result" + until: "heketi_result.stdout | from_json | json_query('items[*]') | length == 0" + retries: 60 + delay: 5 +- name: Tear down glusterfs. + command: "{{ bin_dir }}/kubectl delete daemonset.extensions/glusterfs" + ignore_errors: true # noqa ignore-errors +- name: Remove heketi storage service. + command: "{{ bin_dir }}/kubectl delete service heketi-storage-endpoints" + ignore_errors: true # noqa ignore-errors +- name: Remove heketi gluster role binding + command: "{{ bin_dir }}/kubectl delete clusterrolebinding heketi-gluster-admin" + ignore_errors: true # noqa ignore-errors +- name: Remove heketi config secret + command: "{{ bin_dir }}/kubectl delete secret heketi-config-secret" + ignore_errors: true # noqa ignore-errors +- name: Remove heketi db backup + command: "{{ bin_dir }}/kubectl delete secret heketi-db-backup" + ignore_errors: true # noqa ignore-errors +- name: Remove heketi service account + command: "{{ bin_dir }}/kubectl delete serviceaccount heketi-service-account" + ignore_errors: true # noqa ignore-errors +- name: Get secrets + command: "{{ bin_dir }}/kubectl get secrets --output=\"json\"" + register: "secrets" + changed_when: false +- name: Remove heketi storage secret + vars: { storage_query: "items[?metadata.annotations.\"kubernetes.io/service-account.name\"=='heketi-service-account'].metadata.name|[0]" } + command: "{{ bin_dir }}/kubectl delete secret {{ secrets.stdout | from_json | json_query(storage_query) }}" + when: "storage_query is defined" + ignore_errors: true # noqa ignore-errors diff --git a/kubespray/contrib/offline/README.md b/kubespray/contrib/offline/README.md new file mode 100644 index 0000000..c059f5f --- /dev/null +++ b/kubespray/contrib/offline/README.md @@ -0,0 +1,65 @@ +# Offline deployment + +## manage-offline-container-images.sh + +Container image collecting script for offline deployment + +This script has two features: +(1) Get container images from an environment which is deployed online. +(2) Deploy local container registry and register the container images to the registry. + +Step(1) should be done online site as a preparation, then we bring the gotten images +to the target offline environment. if images are from a private registry, +you need to set `PRIVATE_REGISTRY` environment variable. +Then we will run step(2) for registering the images to local registry. + +Step(1) can be operated with: + +```shell +manage-offline-container-images.sh create +``` + +Step(2) can be operated with: + +```shell +manage-offline-container-images.sh register +``` + +## generate_list.sh + +This script generates the list of downloaded files and the list of container images by `roles/download/defaults/main/main.yml` file. + +Run this script will execute `generate_list.yml` playbook in kubespray root directory and generate four files, +all downloaded files url in files.list, all container images in images.list, jinja2 templates in *.template. + +```shell +./generate_list.sh +tree temp +temp +├── files.list +├── files.list.template +├── images.list +└── images.list.template +0 directories, 5 files +``` + +In some cases you may want to update some component version, you can declare version variables in ansible inventory file or group_vars, +then run `./generate_list.sh -i [inventory_file]` to update file.list and images.list. + +## manage-offline-files.sh + +This script will download all files according to `temp/files.list` and run nginx container to provide offline file download. + +Step(1) generate `files.list` + +```shell +./generate_list.sh +``` + +Step(2) download files and run nginx container + +```shell +./manage-offline-files.sh +``` + +when nginx container is running, it can be accessed through . diff --git a/kubespray/contrib/offline/docker-daemon.json b/kubespray/contrib/offline/docker-daemon.json new file mode 100644 index 0000000..84ddb60 --- /dev/null +++ b/kubespray/contrib/offline/docker-daemon.json @@ -0,0 +1 @@ +{ "insecure-registries":["HOSTNAME:5000"] } diff --git a/kubespray/contrib/offline/generate_list.sh b/kubespray/contrib/offline/generate_list.sh new file mode 100755 index 0000000..646360f --- /dev/null +++ b/kubespray/contrib/offline/generate_list.sh @@ -0,0 +1,33 @@ +#!/bin/bash +set -eo pipefail + +CURRENT_DIR=$(cd $(dirname $0); pwd) +TEMP_DIR="${CURRENT_DIR}/temp" +REPO_ROOT_DIR="${CURRENT_DIR%/contrib/offline}" + +: ${DOWNLOAD_YML:="roles/download/defaults/main/main.yml"} + +mkdir -p ${TEMP_DIR} + +# generate all download files url template +grep 'download_url:' ${REPO_ROOT_DIR}/${DOWNLOAD_YML} \ + | sed 's/^.*_url: //g;s/\"//g' > ${TEMP_DIR}/files.list.template + +# generate all images list template +sed -n '/^downloads:/,/download_defaults:/p' ${REPO_ROOT_DIR}/${DOWNLOAD_YML} \ + | sed -n "s/repo: //p;s/tag: //p" | tr -d ' ' \ + | sed 'N;s#\n# #g' | tr ' ' ':' | sed 's/\"//g' > ${TEMP_DIR}/images.list.template + +# add kube-* images to images list template +# Those container images are downloaded by kubeadm, then roles/download/defaults/main/main.yml +# doesn't contain those images. That is reason why here needs to put those images into the +# list separately. +KUBE_IMAGES="kube-apiserver kube-controller-manager kube-scheduler kube-proxy" +for i in $KUBE_IMAGES; do + echo "{{ kube_image_repo }}/$i:{{ kube_version }}" >> ${TEMP_DIR}/images.list.template +done + +# run ansible to expand templates +/bin/cp ${CURRENT_DIR}/generate_list.yml ${REPO_ROOT_DIR} + +(cd ${REPO_ROOT_DIR} && ansible-playbook $* generate_list.yml && /bin/rm generate_list.yml) || exit 1 diff --git a/kubespray/contrib/offline/generate_list.yml b/kubespray/contrib/offline/generate_list.yml new file mode 100644 index 0000000..bebf349 --- /dev/null +++ b/kubespray/contrib/offline/generate_list.yml @@ -0,0 +1,22 @@ +--- +- name: Collect container images for offline deployment + hosts: localhost + become: no + + roles: + # Just load default variables from roles. + - role: kubespray-defaults + when: false + - role: download + when: false + + tasks: + # Generate files.list and images.list files from templates. + - name: Collect container images for offline deployment + template: + src: ./contrib/offline/temp/{{ item }}.list.template + dest: ./contrib/offline/temp/{{ item }}.list + mode: 0644 + with_items: + - files + - images diff --git a/kubespray/contrib/offline/manage-offline-container-images.sh b/kubespray/contrib/offline/manage-offline-container-images.sh new file mode 100755 index 0000000..40ff2c2 --- /dev/null +++ b/kubespray/contrib/offline/manage-offline-container-images.sh @@ -0,0 +1,172 @@ +#!/bin/bash + +OPTION=$1 +CURRENT_DIR=$(cd $(dirname $0); pwd) +TEMP_DIR="${CURRENT_DIR}/temp" + +IMAGE_TAR_FILE="${CURRENT_DIR}/container-images.tar.gz" +IMAGE_DIR="${CURRENT_DIR}/container-images" +IMAGE_LIST="${IMAGE_DIR}/container-images.txt" +RETRY_COUNT=5 + +function create_container_image_tar() { + set -e + + IMAGES=$(kubectl describe pods --all-namespaces | grep " Image:" | awk '{print $2}' | sort | uniq) + # NOTE: etcd and pause cannot be seen as pods. + # The pause image is used for --pod-infra-container-image option of kubelet. + EXT_IMAGES=$(kubectl cluster-info dump | egrep "quay.io/coreos/etcd:|registry.k8s.io/pause:" | sed s@\"@@g) + IMAGES="${IMAGES} ${EXT_IMAGES}" + + rm -f ${IMAGE_TAR_FILE} + rm -rf ${IMAGE_DIR} + mkdir ${IMAGE_DIR} + cd ${IMAGE_DIR} + + sudo docker pull registry:latest + sudo docker save -o registry-latest.tar registry:latest + + for image in ${IMAGES} + do + FILE_NAME="$(echo ${image} | sed s@"/"@"-"@g | sed s/":"/"-"/g)".tar + set +e + for step in $(seq 1 ${RETRY_COUNT}) + do + sudo docker pull ${image} + if [ $? -eq 0 ]; then + break + fi + echo "Failed to pull ${image} at step ${step}" + if [ ${step} -eq ${RETRY_COUNT} ]; then + exit 1 + fi + done + set -e + sudo docker save -o ${FILE_NAME} ${image} + + # NOTE: Here removes the following repo parts from each image + # so that these parts will be replaced with Kubespray. + # - kube_image_repo: "registry.k8s.io" + # - gcr_image_repo: "gcr.io" + # - docker_image_repo: "docker.io" + # - quay_image_repo: "quay.io" + FIRST_PART=$(echo ${image} | awk -F"/" '{print $1}') + if [ "${FIRST_PART}" = "registry.k8s.io" ] || + [ "${FIRST_PART}" = "gcr.io" ] || + [ "${FIRST_PART}" = "docker.io" ] || + [ "${FIRST_PART}" = "quay.io" ] || + [ "${FIRST_PART}" = "${PRIVATE_REGISTRY}" ]; then + image=$(echo ${image} | sed s@"${FIRST_PART}/"@@) + fi + echo "${FILE_NAME} ${image}" >> ${IMAGE_LIST} + done + + cd .. + sudo chown ${USER} ${IMAGE_DIR}/* + tar -zcvf ${IMAGE_TAR_FILE} ./container-images + rm -rf ${IMAGE_DIR} + + echo "" + echo "${IMAGE_TAR_FILE} is created to contain your container images." + echo "Please keep this file and bring it to your offline environment." +} + +function register_container_images() { + if [ ! -f ${IMAGE_TAR_FILE} ]; then + echo "${IMAGE_TAR_FILE} should exist." + exit 1 + fi + if [ ! -d ${TEMP_DIR} ]; then + mkdir ${TEMP_DIR} + fi + + # To avoid "http: server gave http response to https client" error. + LOCALHOST_NAME=$(hostname) + if [ -d /etc/docker/ ]; then + set -e + # Ubuntu18.04, RHEL7/CentOS7 + cp ${CURRENT_DIR}/docker-daemon.json ${TEMP_DIR}/docker-daemon.json + sed -i s@"HOSTNAME"@"${LOCALHOST_NAME}"@ ${TEMP_DIR}/docker-daemon.json + sudo cp ${TEMP_DIR}/docker-daemon.json /etc/docker/daemon.json + elif [ -d /etc/containers/ ]; then + set -e + # RHEL8/CentOS8 + cp ${CURRENT_DIR}/registries.conf ${TEMP_DIR}/registries.conf + sed -i s@"HOSTNAME"@"${LOCALHOST_NAME}"@ ${TEMP_DIR}/registries.conf + sudo cp ${TEMP_DIR}/registries.conf /etc/containers/registries.conf + else + echo "docker package(docker-ce, etc.) should be installed" + exit 1 + fi + + tar -zxvf ${IMAGE_TAR_FILE} + sudo docker load -i ${IMAGE_DIR}/registry-latest.tar + set +e + sudo docker container inspect registry >/dev/null 2>&1 + if [ $? -ne 0 ]; then + sudo docker run --restart=always -d -p 5000:5000 --name registry registry:latest + fi + set -e + + while read -r line; do + file_name=$(echo ${line} | awk '{print $1}') + raw_image=$(echo ${line} | awk '{print $2}') + new_image="${LOCALHOST_NAME}:5000/${raw_image}" + org_image=$(sudo docker load -i ${IMAGE_DIR}/${file_name} | head -n1 | awk '{print $3}') + image_id=$(sudo docker image inspect ${org_image} | grep "\"Id\":" | awk -F: '{print $3}'| sed s/'\",'//) + if [ -z "${file_name}" ]; then + echo "Failed to get file_name for line ${line}" + exit 1 + fi + if [ -z "${raw_image}" ]; then + echo "Failed to get raw_image for line ${line}" + exit 1 + fi + if [ -z "${org_image}" ]; then + echo "Failed to get org_image for line ${line}" + exit 1 + fi + if [ -z "${image_id}" ]; then + echo "Failed to get image_id for file ${file_name}" + exit 1 + fi + sudo docker load -i ${IMAGE_DIR}/${file_name} + sudo docker tag ${image_id} ${new_image} + sudo docker push ${new_image} + done <<< "$(cat ${IMAGE_LIST})" + + echo "Succeeded to register container images to local registry." + echo "Please specify ${LOCALHOST_NAME}:5000 for the following options in your inventry:" + echo "- kube_image_repo" + echo "- gcr_image_repo" + echo "- docker_image_repo" + echo "- quay_image_repo" +} + +if [ "${OPTION}" == "create" ]; then + create_container_image_tar +elif [ "${OPTION}" == "register" ]; then + register_container_images +else + echo "This script has two features:" + echo "(1) Get container images from an environment which is deployed online." + echo "(2) Deploy local container registry and register the container images to the registry." + echo "" + echo "Step(1) should be done online site as a preparation, then we bring" + echo "the gotten images to the target offline environment. if images are from" + echo "a private registry, you need to set PRIVATE_REGISTRY environment variable." + echo "Then we will run step(2) for registering the images to local registry." + echo "" + echo "${IMAGE_TAR_FILE} is created to contain your container images." + echo "Please keep this file and bring it to your offline environment." + echo "" + echo "Step(1) can be operated with:" + echo " $ ./manage-offline-container-images.sh create" + echo "" + echo "Step(2) can be operated with:" + echo " $ ./manage-offline-container-images.sh register" + echo "" + echo "Please specify 'create' or 'register'." + echo "" + exit 1 +fi diff --git a/kubespray/contrib/offline/manage-offline-files.sh b/kubespray/contrib/offline/manage-offline-files.sh new file mode 100755 index 0000000..41936cd --- /dev/null +++ b/kubespray/contrib/offline/manage-offline-files.sh @@ -0,0 +1,44 @@ +#!/bin/bash + +CURRENT_DIR=$( dirname "$(readlink -f "$0")" ) +OFFLINE_FILES_DIR_NAME="offline-files" +OFFLINE_FILES_DIR="${CURRENT_DIR}/${OFFLINE_FILES_DIR_NAME}" +OFFLINE_FILES_ARCHIVE="${CURRENT_DIR}/offline-files.tar.gz" +FILES_LIST=${FILES_LIST:-"${CURRENT_DIR}/temp/files.list"} +NGINX_PORT=8080 + +# download files +if [ ! -f "${FILES_LIST}" ]; then + echo "${FILES_LIST} should exist, run ./generate_list.sh first." + exit 1 +fi + +rm -rf "${OFFLINE_FILES_DIR}" +rm "${OFFLINE_FILES_ARCHIVE}" +mkdir "${OFFLINE_FILES_DIR}" + +wget -x -P "${OFFLINE_FILES_DIR}" -i "${FILES_LIST}" +tar -czvf "${OFFLINE_FILES_ARCHIVE}" "${OFFLINE_FILES_DIR_NAME}" + +[ -n "$NO_HTTP_SERVER" ] && echo "skip to run nginx" && exit 0 + +# run nginx container server +if command -v nerdctl 1>/dev/null 2>&1; then + runtime="nerdctl" +elif command -v podman 1>/dev/null 2>&1; then + runtime="podman" +elif command -v docker 1>/dev/null 2>&1; then + runtime="docker" +else + echo "No supported container runtime found" + exit 1 +fi + +sudo "${runtime}" container inspect nginx >/dev/null 2>&1 +if [ $? -ne 0 ]; then + sudo "${runtime}" run \ + --restart=always -d -p ${NGINX_PORT}:80 \ + --volume "${OFFLINE_FILES_DIR}:/usr/share/nginx/html/download" \ + --volume "${CURRENT_DIR}"/nginx.conf:/etc/nginx/nginx.conf \ + --name nginx nginx:alpine +fi diff --git a/kubespray/contrib/offline/nginx.conf b/kubespray/contrib/offline/nginx.conf new file mode 100644 index 0000000..a6fd5eb --- /dev/null +++ b/kubespray/contrib/offline/nginx.conf @@ -0,0 +1,39 @@ +user nginx; +worker_processes auto; +error_log /var/log/nginx/error.log; +pid /run/nginx.pid; +include /usr/share/nginx/modules/*.conf; +events { + worker_connections 1024; +} +http { + log_format main '$remote_addr - $remote_user [$time_local] "$request" ' + '$status $body_bytes_sent "$http_referer" ' + '"$http_user_agent" "$http_x_forwarded_for"'; + access_log /var/log/nginx/access.log main; + sendfile on; + tcp_nopush on; + tcp_nodelay on; + keepalive_timeout 65; + types_hash_max_size 2048; + default_type application/octet-stream; + include /etc/nginx/conf.d/*.conf; + server { + listen 80 default_server; + listen [::]:80 default_server; + server_name _; + include /etc/nginx/default.d/*.conf; + location / { + root /usr/share/nginx/html/download; + autoindex on; + autoindex_exact_size off; + autoindex_localtime on; + } + error_page 404 /404.html; + location = /40x.html { + } + error_page 500 502 503 504 /50x.html; + location = /50x.html { + } + } +} diff --git a/kubespray/contrib/offline/registries.conf b/kubespray/contrib/offline/registries.conf new file mode 100644 index 0000000..6852aee --- /dev/null +++ b/kubespray/contrib/offline/registries.conf @@ -0,0 +1,8 @@ +[registries.search] +registries = ['registry.access.redhat.com', 'registry.redhat.io', 'docker.io'] + +[registries.insecure] +registries = ['HOSTNAME:5000'] + +[registries.block] +registries = [] diff --git a/kubespray/contrib/os-services/os-services.yml b/kubespray/contrib/os-services/os-services.yml new file mode 100644 index 0000000..eb120d4 --- /dev/null +++ b/kubespray/contrib/os-services/os-services.yml @@ -0,0 +1,5 @@ +--- +- name: Disable firewalld/ufw + hosts: all + roles: + - { role: prepare } diff --git a/kubespray/contrib/os-services/roles/prepare/defaults/main.yml b/kubespray/contrib/os-services/roles/prepare/defaults/main.yml new file mode 100644 index 0000000..9c4a149 --- /dev/null +++ b/kubespray/contrib/os-services/roles/prepare/defaults/main.yml @@ -0,0 +1,2 @@ +--- +disable_service_firewall: false diff --git a/kubespray/contrib/os-services/roles/prepare/tasks/main.yml b/kubespray/contrib/os-services/roles/prepare/tasks/main.yml new file mode 100644 index 0000000..e95dcef --- /dev/null +++ b/kubespray/contrib/os-services/roles/prepare/tasks/main.yml @@ -0,0 +1,23 @@ +--- +- name: Disable firewalld and ufw + when: + - disable_service_firewall is defined and disable_service_firewall + block: + - name: List services + service_facts: + + - name: Disable service firewalld + systemd: + name: firewalld + state: stopped + enabled: no + when: + "'firewalld.service' in services and services['firewalld.service'].status != 'not-found'" + + - name: Disable service ufw + systemd: + name: ufw + state: stopped + enabled: no + when: + "'ufw.service' in services and services['ufw.service'].status != 'not-found'" diff --git a/kubespray/contrib/packaging/rpm/kubespray.spec b/kubespray/contrib/packaging/rpm/kubespray.spec new file mode 100644 index 0000000..cc8a752 --- /dev/null +++ b/kubespray/contrib/packaging/rpm/kubespray.spec @@ -0,0 +1,62 @@ +%global srcname kubespray + +%{!?upstream_version: %global upstream_version %{version}%{?milestone}} + +Name: kubespray +Version: master +Release: %(git describe | sed -r 's/v(\S+-?)-(\S+)-(\S+)/\1.dev\2+\3/') +Summary: Ansible modules for installing Kubernetes + +Group: System Environment/Libraries +License: ASL 2.0 +Url: https://github.com/kubernetes-sigs/kubespray +Source0: https://github.com/kubernetes-sigs/kubespray/archive/%{upstream_version}.tar.gz#/%{name}-%{release}.tar.gz + +BuildArch: noarch +BuildRequires: git +BuildRequires: python2 +BuildRequires: python2-devel +BuildRequires: python2-setuptools +BuildRequires: python-d2to1 +BuildRequires: python2-pbr + +Requires: ansible >= 2.5.0 +Requires: python-jinja2 >= 2.10 +Requires: python-netaddr +Requires: python-pbr + +%description + +Ansible-kubespray is a set of Ansible modules and playbooks for +installing a Kubernetes cluster. If you have questions, join us +on the https://slack.k8s.io, channel '#kubespray'. + +%prep +%autosetup -n %{name}-%{upstream_version} -S git + + +%build +export PBR_VERSION=%{release} +%{__python2} setup.py build bdist_rpm + + +%install +export PBR_VERSION=%{release} +export SKIP_PIP_INSTALL=1 +%{__python2} setup.py install --skip-build --root %{buildroot} bdist_rpm + + +%files +%doc %{_docdir}/%{name}/README.md +%doc %{_docdir}/%{name}/inventory/sample/hosts.ini +%config %{_sysconfdir}/%{name}/ansible.cfg +%config %{_sysconfdir}/%{name}/inventory/sample/group_vars/all.yml +%config %{_sysconfdir}/%{name}/inventory/sample/group_vars/k8s_cluster.yml +%license %{_docdir}/%{name}/LICENSE +%{python2_sitelib}/%{srcname}-%{release}-py%{python2_version}.egg-info +%{_datarootdir}/%{name}/roles/ +%{_datarootdir}/%{name}/playbooks/ +%defattr(-,root,root) + + +%changelog diff --git a/kubespray/contrib/terraform/OWNERS b/kubespray/contrib/terraform/OWNERS new file mode 100644 index 0000000..b58878d --- /dev/null +++ b/kubespray/contrib/terraform/OWNERS @@ -0,0 +1,5 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +approvers: + - holmsten + - miouge1 diff --git a/kubespray/contrib/terraform/aws/.gitignore b/kubespray/contrib/terraform/aws/.gitignore new file mode 100644 index 0000000..373687b --- /dev/null +++ b/kubespray/contrib/terraform/aws/.gitignore @@ -0,0 +1,3 @@ +*.tfstate* +.terraform.lock.hcl +.terraform diff --git a/kubespray/contrib/terraform/aws/README.md b/kubespray/contrib/terraform/aws/README.md new file mode 100644 index 0000000..7e3428d --- /dev/null +++ b/kubespray/contrib/terraform/aws/README.md @@ -0,0 +1,162 @@ +# Kubernetes on AWS with Terraform + +## Overview + +This project will create: + +- VPC with Public and Private Subnets in # Availability Zones +- Bastion Hosts and NAT Gateways in the Public Subnet +- A dynamic number of masters, etcd, and worker nodes in the Private Subnet + - even distributed over the # of Availability Zones +- AWS ELB in the Public Subnet for accessing the Kubernetes API from the internet + +## Requirements + +- Terraform 0.12.0 or newer + +## How to Use + +- Export the variables for your AWS credentials or edit `credentials.tfvars`: + +```commandline +export TF_VAR_AWS_ACCESS_KEY_ID="www" +export TF_VAR_AWS_SECRET_ACCESS_KEY ="xxx" +export TF_VAR_AWS_SSH_KEY_NAME="yyy" +export TF_VAR_AWS_DEFAULT_REGION="zzz" +``` + +- Update `contrib/terraform/aws/terraform.tfvars` with your data. By default, the Terraform scripts use Ubuntu 18.04 LTS (Bionic) as base image. If you want to change this behaviour, see note "Using other distrib than Ubuntu" below. +- Create an AWS EC2 SSH Key +- Run with `terraform apply --var-file="credentials.tfvars"` or `terraform apply` depending if you exported your AWS credentials + +Example: + +```commandline +terraform apply -var-file=credentials.tfvars +``` + +- Terraform automatically creates an Ansible Inventory file called `hosts` with the created infrastructure in the directory `inventory` +- Ansible will automatically generate an ssh config file for your bastion hosts. To connect to hosts with ssh using bastion host use generated `ssh-bastion.conf`. Ansible automatically detects bastion and changes `ssh_args` + +```commandline +ssh -F ./ssh-bastion.conf user@$ip +``` + +- Once the infrastructure is created, you can run the kubespray playbooks and supply inventory/hosts with the `-i` flag. + +Example (this one assumes you are using Ubuntu) + +```commandline +ansible-playbook -i ./inventory/hosts ./cluster.yml -e ansible_user=ubuntu -b --become-user=root --flush-cache +``` + +***Using other distrib than Ubuntu*** +If you want to use another distribution than Ubuntu 18.04 (Bionic) LTS, you can modify the search filters of the 'data "aws_ami" "distro"' in variables.tf. + +For example, to use: + +- Debian Jessie, replace 'data "aws_ami" "distro"' in variables.tf with + +```ini +data "aws_ami" "distro" { + most_recent = true + + filter { + name = "name" + values = ["debian-jessie-amd64-hvm-*"] + } + + filter { + name = "virtualization-type" + values = ["hvm"] + } + + owners = ["379101102735"] +} +``` + +- Ubuntu 16.04, replace 'data "aws_ami" "distro"' in variables.tf with + +```ini +data "aws_ami" "distro" { + most_recent = true + + filter { + name = "name" + values = ["ubuntu/images/hvm-ssd/ubuntu-xenial-16.04-amd64-*"] + } + + filter { + name = "virtualization-type" + values = ["hvm"] + } + + owners = ["099720109477"] +} +``` + +- Centos 7, replace 'data "aws_ami" "distro"' in variables.tf with + +```ini +data "aws_ami" "distro" { + most_recent = true + + filter { + name = "name" + values = ["dcos-centos7-*"] + } + + filter { + name = "virtualization-type" + values = ["hvm"] + } + + owners = ["688023202711"] +} +``` + +## Connecting to Kubernetes + +You can use the following set of commands to get the kubeconfig file from your newly created cluster. Before running the commands, make sure you are in the project's root folder. + +```commandline +# Get the controller's IP address. +CONTROLLER_HOST_NAME=$(cat ./inventory/hosts | grep "\[kube_control_plane\]" -A 1 | tail -n 1) +CONTROLLER_IP=$(cat ./inventory/hosts | grep $CONTROLLER_HOST_NAME | grep ansible_host | cut -d'=' -f2) + +# Get the hostname of the load balancer. +LB_HOST=$(cat inventory/hosts | grep apiserver_loadbalancer_domain_name | cut -d'"' -f2) + +# Get the controller's SSH fingerprint. +ssh-keygen -R $CONTROLLER_IP > /dev/null 2>&1 +ssh-keyscan -H $CONTROLLER_IP >> ~/.ssh/known_hosts 2>/dev/null + +# Get the kubeconfig from the controller. +mkdir -p ~/.kube +ssh -F ssh-bastion.conf centos@$CONTROLLER_IP "sudo chmod 644 /etc/kubernetes/admin.conf" +scp -F ssh-bastion.conf centos@$CONTROLLER_IP:/etc/kubernetes/admin.conf ~/.kube/config +sed -i "s^server:.*^server: https://$LB_HOST:6443^" ~/.kube/config +kubectl get nodes +``` + +## Troubleshooting + +### Remaining AWS IAM Instance Profile + +If the cluster was destroyed without using Terraform it is possible that +the AWS IAM Instance Profiles still remain. To delete them you can use +the `AWS CLI` with the following command: + +```commandline +aws iam delete-instance-profile --region --instance-profile-name +``` + +### Ansible Inventory doesn't get created + +It could happen that Terraform doesn't create an Ansible Inventory file automatically. If this is the case copy the output after `inventory=` and create a file named `hosts`in the directory `inventory` and paste the inventory into the file. + +## Architecture + +Pictured is an AWS Infrastructure created with this Terraform project distributed over two Availability Zones. + +![AWS Infrastructure with Terraform ](docs/aws_kubespray.png) diff --git a/kubespray/contrib/terraform/aws/create-infrastructure.tf b/kubespray/contrib/terraform/aws/create-infrastructure.tf new file mode 100644 index 0000000..0a38844 --- /dev/null +++ b/kubespray/contrib/terraform/aws/create-infrastructure.tf @@ -0,0 +1,179 @@ +terraform { + required_version = ">= 0.12.0" +} + +provider "aws" { + access_key = var.AWS_ACCESS_KEY_ID + secret_key = var.AWS_SECRET_ACCESS_KEY + region = var.AWS_DEFAULT_REGION +} + +data "aws_availability_zones" "available" {} + +/* +* Calling modules who create the initial AWS VPC / AWS ELB +* and AWS IAM Roles for Kubernetes Deployment +*/ + +module "aws-vpc" { + source = "./modules/vpc" + + aws_cluster_name = var.aws_cluster_name + aws_vpc_cidr_block = var.aws_vpc_cidr_block + aws_avail_zones = data.aws_availability_zones.available.names + aws_cidr_subnets_private = var.aws_cidr_subnets_private + aws_cidr_subnets_public = var.aws_cidr_subnets_public + default_tags = var.default_tags +} + +module "aws-nlb" { + source = "./modules/nlb" + + aws_cluster_name = var.aws_cluster_name + aws_vpc_id = module.aws-vpc.aws_vpc_id + aws_avail_zones = data.aws_availability_zones.available.names + aws_subnet_ids_public = module.aws-vpc.aws_subnet_ids_public + aws_nlb_api_port = var.aws_nlb_api_port + k8s_secure_api_port = var.k8s_secure_api_port + default_tags = var.default_tags +} + +module "aws-iam" { + source = "./modules/iam" + + aws_cluster_name = var.aws_cluster_name +} + +/* +* Create Bastion Instances in AWS +* +*/ + +resource "aws_instance" "bastion-server" { + ami = data.aws_ami.distro.id + instance_type = var.aws_bastion_size + count = var.aws_bastion_num + associate_public_ip_address = true + subnet_id = element(module.aws-vpc.aws_subnet_ids_public, count.index) + + vpc_security_group_ids = module.aws-vpc.aws_security_group + + key_name = var.AWS_SSH_KEY_NAME + + tags = merge(var.default_tags, tomap({ + Name = "kubernetes-${var.aws_cluster_name}-bastion-${count.index}" + Cluster = var.aws_cluster_name + Role = "bastion-${var.aws_cluster_name}-${count.index}" + })) +} + +/* +* Create K8s Master and worker nodes and etcd instances +* +*/ + +resource "aws_instance" "k8s-master" { + ami = data.aws_ami.distro.id + instance_type = var.aws_kube_master_size + + count = var.aws_kube_master_num + + subnet_id = element(module.aws-vpc.aws_subnet_ids_private, count.index) + + vpc_security_group_ids = module.aws-vpc.aws_security_group + + root_block_device { + volume_size = var.aws_kube_master_disk_size + } + + iam_instance_profile = module.aws-iam.kube_control_plane-profile + key_name = var.AWS_SSH_KEY_NAME + + tags = merge(var.default_tags, tomap({ + Name = "kubernetes-${var.aws_cluster_name}-master${count.index}" + "kubernetes.io/cluster/${var.aws_cluster_name}" = "member" + Role = "master" + })) +} + +resource "aws_lb_target_group_attachment" "tg-attach_master_nodes" { + count = var.aws_kube_master_num + target_group_arn = module.aws-nlb.aws_nlb_api_tg_arn + target_id = element(aws_instance.k8s-master.*.private_ip, count.index) +} + +resource "aws_instance" "k8s-etcd" { + ami = data.aws_ami.distro.id + instance_type = var.aws_etcd_size + + count = var.aws_etcd_num + + subnet_id = element(module.aws-vpc.aws_subnet_ids_private, count.index) + + vpc_security_group_ids = module.aws-vpc.aws_security_group + + root_block_device { + volume_size = var.aws_etcd_disk_size + } + + key_name = var.AWS_SSH_KEY_NAME + + tags = merge(var.default_tags, tomap({ + Name = "kubernetes-${var.aws_cluster_name}-etcd${count.index}" + "kubernetes.io/cluster/${var.aws_cluster_name}" = "member" + Role = "etcd" + })) +} + +resource "aws_instance" "k8s-worker" { + ami = data.aws_ami.distro.id + instance_type = var.aws_kube_worker_size + + count = var.aws_kube_worker_num + + subnet_id = element(module.aws-vpc.aws_subnet_ids_private, count.index) + + vpc_security_group_ids = module.aws-vpc.aws_security_group + + root_block_device { + volume_size = var.aws_kube_worker_disk_size + } + + iam_instance_profile = module.aws-iam.kube-worker-profile + key_name = var.AWS_SSH_KEY_NAME + + tags = merge(var.default_tags, tomap({ + Name = "kubernetes-${var.aws_cluster_name}-worker${count.index}" + "kubernetes.io/cluster/${var.aws_cluster_name}" = "member" + Role = "worker" + })) +} + +/* +* Create Kubespray Inventory File +* +*/ +data "template_file" "inventory" { + template = file("${path.module}/templates/inventory.tpl") + + vars = { + public_ip_address_bastion = join("\n", formatlist("bastion ansible_host=%s", aws_instance.bastion-server.*.public_ip)) + connection_strings_master = join("\n", formatlist("%s ansible_host=%s", aws_instance.k8s-master.*.private_dns, aws_instance.k8s-master.*.private_ip)) + connection_strings_node = join("\n", formatlist("%s ansible_host=%s", aws_instance.k8s-worker.*.private_dns, aws_instance.k8s-worker.*.private_ip)) + list_master = join("\n", aws_instance.k8s-master.*.private_dns) + list_node = join("\n", aws_instance.k8s-worker.*.private_dns) + connection_strings_etcd = join("\n", formatlist("%s ansible_host=%s", aws_instance.k8s-etcd.*.private_dns, aws_instance.k8s-etcd.*.private_ip)) + list_etcd = join("\n", ((var.aws_etcd_num > 0) ? (aws_instance.k8s-etcd.*.private_dns) : (aws_instance.k8s-master.*.private_dns))) + nlb_api_fqdn = "apiserver_loadbalancer_domain_name=\"${module.aws-nlb.aws_nlb_api_fqdn}\"" + } +} + +resource "null_resource" "inventories" { + provisioner "local-exec" { + command = "echo '${data.template_file.inventory.rendered}' > ${var.inventory_file}" + } + + triggers = { + template = data.template_file.inventory.rendered + } +} diff --git a/kubespray/contrib/terraform/aws/credentials.tfvars.example b/kubespray/contrib/terraform/aws/credentials.tfvars.example new file mode 100644 index 0000000..19420c5 --- /dev/null +++ b/kubespray/contrib/terraform/aws/credentials.tfvars.example @@ -0,0 +1,8 @@ +#AWS Access Key +AWS_ACCESS_KEY_ID = "" +#AWS Secret Key +AWS_SECRET_ACCESS_KEY = "" +#EC2 SSH Key Name +AWS_SSH_KEY_NAME = "" +#AWS Region +AWS_DEFAULT_REGION = "eu-central-1" diff --git a/kubespray/contrib/terraform/aws/docs/aws_kubespray.png b/kubespray/contrib/terraform/aws/docs/aws_kubespray.png new file mode 100644 index 0000000000000000000000000000000000000000..40245b845a5094d3cef4500d4f256031329a7b2e GIT binary patch literal 116455 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%0AOMpYg_>;9sAdO zpHOT8-trMfEmt)TV`^;nqUiZKLYC~rfFAlY!Rp0s(;1jB zKFDDKwIckobCL;pwlOt9r*^f2&CD`tBmZ>wu)xFNjl4%=_p#P;=b4g5& ze{Jqn`IGr8F-~+5KuVC*W+4=p$>>Hw@aPL`QR6z_c>h{N)5T$==b^g6_2wCfk zs%soCwgD2L_w92lkTra##aR{C0_-tQMupLg3vwFt_`QnekR~Woj;^DVP>DTwjmLy zI|Pr(wwVaHXJcb_rg4;YwxG>nd-Vz7_9{Xg#nKW(zO32${b@rFBr+%DKCT%-GbuiQ3A-OLoN55#aO9I${F@)id zX&||Blsh+g_EzX`UIXZ`@O;SQ6u@{!(7PxUg-q6AqZQcK4{fnF^@XHoZM!ert)06=3W43c{x2~$7et0!kO47d-$6V?Y1DL|>Z<{U zXfN=WZuf?WB(^;H4g$gp=SlvEBqf12MW6+5iG%-lPUXjFNQ4g}5mzVq1)f`Y(>iVa z<23mrUtsVX5pLFZzKz(Az-+8;IobbahKNaRvg&z1`+WR$>ejlp70+&bgKs&yC5x9Q%WEbO znhBqq!ZH#pT-XMa$!NB^KD42qrSuim1fTpwl5Z+4x=s~CLncIHJdsH=+nyx7^mOS) zE%1TG{oCyXQQ^5>NZjj$z_}l8t&5kRyVyKjVrlgj4F|t#S3}wc8gK3O2nRh~{9E}y zVHt<(<$qFcS?y$3J9GEw=Sq{gD0^rN?~@Y(PHz!xGcXYjK=5?y3h5vncWFrmw>!sx zf)Xm+TV7A37%{!FD)Pnb&2K|$;{2K8CgAIU?7tPgM0tH$}zn zxiv4Ev8M~8F)VoZD>q6J*B?v;Eo`f&myfMG^*OuOe+K@%FLL4H-%E4zFKMEZ5Xgf*GM?*izjCyd z6~rNq8xVBLcrriaBgD;mBqU}Z+)sf>m-JL0_?+mE3~MFVd0SPYvW#d z|HzzGe&%H`@YJGbnr&Qieb-N-whWgJ@LB;dtG)v5++!6zs}(t+Q6spR`?Q&Rm!kBL z(MWf^|5kZEed}>LliefCbA&8!cDQNNBAy~Nou_t$7dx54G1A{j7G05oM#kxBo?jOx7{l4b6#NaUElv!AthJI`?_S$@a z$MoCfaNMclPJsZu74?y6#DF06O#BZiH)8% zHQ(+|{=`SC;6R#H=Qnu2HAUf{D6$;>4OOHkC@|xJuH{#c!lot4b~pbs+0oyoqalR} z_~4)ItaZL4E$1GG&$iuC1zw-v#I%zLSlHX7z`K7m9Fa~p93^Ac%ERDSI#D>#B~EQ@ ze(CcqJ><~FVC*cT&J&Ucmo|!8TI-DTPi3TDC>4%EVN&JB5F$kxWAeKgxT5Xg()PW$ zOU|TGowUx8)$GAeJD<&qa<$ZT(f8~sZA)<(1J zw)kd9*gt8`{3YcLqpm>TV`<|CH#^eOnWvnf(fwz6Zresxc#hTK{c5+8^QMeE<8OibEQi%p!1{I!TGqeli5qX zC~os5-cQQ_R}7{o`s*o+C=@K2QQ7w~svH*7?u)4JKzalBrtFpH2{P{}r=g7OQpjj5e^~I$%W^&Sa;jl{>}JWMdrh#(_+++)#m#Z%9Obih z%zmc!9!``OnU+dsLlr|xa$I?D-h#s<7S2X!`I8qM1Xs;PE?1oAOv`i2)&_^_5kE;{+tW-=qYumlgfWjX zH#R*C3tecmSUp`w4Bac_o?J`haJ-d69!4HU8ZHP!oenGUQBHz zU|;O!B)>$~%etd~2CIJULVPL359gz>YkZ{0FaNML-c|igX8t#&9}Ag3&nMmm%tL5lYjcj!O%7J zC#(GjT}^*XMP_i{qRqIev_655mW7DLH6rfii7S_yn|qci1C1+xf@Mqwr=pKQrt4YP>6Wr=@9FRIpDX@D8YWIXS&lxBC6)K62%9n{Tu z`^SPz=Uq1sEz=)OY}06Dh_sp-Vj^Xu`tY$+Vg5oXc!vN*&(5hKxN0r!Ga(!c;aDby z)aWK_rk-6&B;2yi@ww_B4S75`r&vsAcVg5Rvj}rtKow|hn+odAYc*2lo2=f=s4HJf z*P#_Uq=ALLBr4Uo=pz}yl%72|$I3kH7jB#BCQ7J~Lq)%x^wlu%M&QBWy77+-{UrG} zXdTN?j=yG>yy|My{$pP>kTcBl=@>t{U(JER8cg{?J^d{SzN!AW8C_U|NoQaH`M0vt zGYcx@PBEj|)fr2T6az%o#EGS{KyyaDo# z&Y>1HHzVjxccB%S2uy^uP#T6>!@f1>+x_V9*DoY1(xzA9W%ygtYC%!tHjQhgRlRK& z?|TFrp7l-$AQ`l)veMUeMMSr9pJ_|X?4JArV3_{2QMlkXSZ8?g--97WK!X;wR2RO`J;D|W?EX!dGjsi zhIV6@X10CH`6g+?Vb?#Wb|SYp)t=kl2bc`cipX=B))vZ=T6H- z9D<5~$LndKKi+gb{2t0vQ80~Rj^F?Ovhw(1%6kAY;B94Kvh-^3Wb06$^{2kXa@`B_ zX}hDLN2sCGQ6JTnsW40uty!S_>uWV_zoE>$Ax$z(@Ybq;ALy!D?tdO0RpaGz!Q}SF zoDF@H8aV)6bJaF{lp!K~D2K}?B7%yJ6!k9XiQX@WDc0h6?h!GsINkH<8;=+n(-RK8Woc|wk-A$F%fSrP{FNqfIURXs>! z*Mz+L$bz|{{@d}wZsJ5qDBzVbo7|*qcdgtLIeQ~XBRsZ;^=N$TjIr{z>6|^04UX%- zkRs2f9{vrhDm?}3UyqHR-G52iC`P_D3 z#5vO3df6E4G-@=`Y*A?C0;5)F>n<+J8=qZyOO8CT}Zp(f^b z%J|KN$N0Yiwb5%ND@gYJ04VhHm>zDsObW(wcng-d0~9 z1JNS?(#m2&vj6}wqMzMF->26!hJ{zgvj!iJ66q;okbh?l+Ng+T=CM@q^gQv-rZaz5 z{i})MN1;t@9=M?dRsG8070wFkmp(b=4exkspJHxE-_uX4407**O|lFe9LgQDdV1KstpSy9qa#>e%DHvM^d z6+P<{vd@5Q6#rAUdmCHaLh9Lj#nNn{dpk?_C3%a8_jI5IAICnuf9z+vm1cAE?zhL5 zNk^EpOiH#PZ`E>8?87l~P%|?kxc*CvYflzkMba}Phx>EBC$4ip2%izo7+I*h;t3D? zSwKa~QTM}fWuZ7ii;%Ttu5o|N^28c*`k!MH-x$v+ehi2f z!l=9E$6kleK}D(d5YO|VkfxT32uk#b*KFQIS}kXdzyl=r?=$)z!Z5a5d8Z&AS8;4s zukq{uN&JpBn4j{%Ptzd* z)n`!&I3W8V;DFE^F){Lh1Cr;Gp2Uy?%p7yVNNmS?(W!l?9XBb_rav~7-*b?Q-`jBE zn=i3YwN8Kb4EEXO*xN@MIB9eo9!n38=x5AE1^PuqmQ_d=Y3+;dee8v;v>MHdib@I^ z5S()$RdbXYdU}DXBx9tb10Yu4x;wzye9@QuX%hgfPd|Z&c_Fi;xBN`_oTcqF05>`H z`oi|yc?4H3Y(EYCq#QreHz$6$@gCWcdJV_-hy=S04#}_aK^B z_kMksh~uogh=1{h$Mia!`)HP|2o-6M4+zGn^CXFwzFa^!`!I90j-p;=ym2VKu$vfKCT1q~(-QL&q~G5$UAk z-0V>Rr^$9P*n^KrA3RfpzXS_`EM^xhEa-SJL%1k|T*w`4Kc)I+2#% z`}y^yDR8G&=TnAkv;KS}qh4Udk6;8qt>AwtCxDD?$e-XzfNA^lzy17*K`)wgoOD## zw~-EI>|`mQOzS@+f)(Tix`DY@aL^f8jTd2)3wXbCHX*LOp!41S;aLSX=iqAzNDWP> z``wu`#H2k>_n?NAn!^3)TwwiGZn$>5Bw#p(hx;b>|1$ss5hP1~cKseU!C^FbiolVo)jd z4K=X$z*a>7;PQ?C8{q!KsD+SbF=vt7Yk{4+jTd2&)_JiE-Xfx2qvCC27U&o0T>}my zEKauFkMd!&&{12bF+eq;x-zSwMdZzF3bw6qiv88>)xaYW!g>jj<#-vcxsA$jRu=sf zJh-KMfd+%0h>oA+J@sgk?|V@zrf|+_e$>zuZi>AH&oTdYmXbGo$O)y80$auj=@FhMtD(s@q0YG=u1Lqsi5&J_<}7JFF@}DYjj=!Ia!(N2#VYZ zhDmW~Z&-lqE3jLu#?Q{6IJMfsG$+;C!^seyx3wz*{=Ssw9xSlZ^PjtRw^;>l_jWS- z{(n>O_>UU^hxF9@GrXSHix|q-}R@nef~}Mq}B9BLc<#HMt_K(FWlCiY&3u z0Ec`n!Jt1u4{{l}QPsWePt44Alw2%Coe3{NpTVMs(fR3}f7;s^Ux$!hBKO|ffmg(h z2Lh|B1HRoh>4@U|31v&2cr?E7++;HJh`^!@cN7S&ZU7<-Oy~Ledu@v{SKjd@^zt#h zGs1 zUkBTRcz}4|eOnlY`m7KWGDXoMqkI+eG5n`piSO2Lhm+lg)7nP&wzKU?Gv}*~?F?;` zY#GBAaljUxSia?@+%!lq7s7!o?_gY+6Pwgc73cRwlr=Y%TyPQzbO6%1dIEhw&S^Fu z8XUfy{~mM0^Ri2|=nifH{QYwAT^Ql<)x-% zvJO5L&)z4dAIqXSX$J~6vX~4{l z9Dw7$M>Ov#{Hq1ZO{($rnynCAz2!LnmJGvpvfiGhus1jWs58J?v9xTUT$!q~$SQ`2 zJ4iQ79#2xUN>NDO06sTDo^nBRz`FQFJ*j$F8TWwTE@k2G=4nmr=6TG2mj~mRM)L!) zFHt`?rX~=9a{TkiB!+YBB!;Fjcbf+Y?GrO}aRH{BRJELxEm7&<6Q(Oeeb@u{@lbf!u@aD!G$Y{+1wk+_a#cPsqW^kS1&REk=@i`}7 zF+c#k1W4P$Ju@<*$p`!oU5uIICE$N{#cwK4aYhl^ZbEXlFB-LDBnfYT;XCYqoO6^3L}MO8ZsB zSp@}g8>K`}7FNs*X@US<{&D2O&pCObgyq^dx6RQyO&8ZURCvi1iL?f?^R_4&aj#Ls zEqDqf1{dq{|94#fo2FJc18FKRdmCXks}v>B%6)EOxNoD;=?gqx9 zOuLkBu1ptqmJE_>I3Gu@$%gAIRO10X^Y@DE6Sb&@>>lcu?ig&?iP&K!7r_~l5ku6r zMqCG!^pzB%aYP+l!^nAfwloe*bR!Al^i z6mdsiJj^!HS6crgzmyjfklZcW-ra76@Psna8IY9I#EumXp-hqrf$pdXy@fn&N2g+ZsG ztI5k~k!9VlKDnP-WrF!0p3@z0P@djk@iE<~p^W#Y8P1? zo4%=&RwJ6Q4@EgP{rz^;0!R#9D2@%h<(L?FkY@ZU6I+=)SiYXi*pQQ|9N}kl3|GID zd+}*m)YFWv@(CRi4gEQmHP?aBp93%S*z9+myAEI^lO;jOTTVYc`A37oluXH}&P|~m zn0`Bt)_EQtr3_+D;-jG^4wFM(D39bdzeWdwc=MN(uyv-NVfON&wq=lEr>>m9mG@%F zuPmz_ir2~MUS7z3WC486W#)hT96^sX48aOo)~D9JI&WU?bUBL2lnZ(l_QR%=u1$yd zfdk1eMO{`A7|2BSRDJRC99BPtK+=y{j61&-)99uBZTmbGYD=Q(nIO`PZLRve^o^oB zW<6$<{wj^hA2iC9e&mWrBs#S zU`NfB60i1@%vKZ`<{cq4$*Qtd2wG|*o6mm#5N3|V6c&7p%Z=S{&qBLGmUe`}7mlA@ z&q@ey!H0j(ntqNx%^8Dvjy^;_{FEuuGh_I3ySm>vK3hiAQ`t4#yd!=wn)!k281#4j z3Z7j`H#ltsxLUlNer1($`mp*Swx4*<63Ab&2h%6^xmL5sB7AYjC>Irfr?$UR+>gdC z#fF7$Cz7ucndxY)J5lp;r0wHM5YoOsVqCd+L~rvUa=C^h*p&`2F{*}*H%XqdncT@U zc=IbkLFF5mfi=7DQpz_9%AG?$g@pyBY?8|nt60;B!!KHT@5)RR0015ZYrLTaP+oCX zQA|xLSK`+t`9VTrrZdxg6(iKU{8Gz{^^IEZp`{Wtl>vh7kC=Qak~!KAdNbK}EHAin zUvN;?Q+&7%InIVU{0 z;4_Ko$Dk7Tqlw;%4)hjW^61Snv(lA&Q;8b+m5`=Ou?Pzlb7SHcy-xA?$xmcSo_%<6-d-XbQ7B(!Zvuq98ELiib^MwFX&i;ltorOmd^kmT zEChA23x@uA@Cny;oF`n2*qwrpjt;dkg2hS}&CmahF)Wx;PUwpCWem27?c{Zgbja(k znqp|zyaungN(6h;;rNu;4i~cBgk-tN;t>Um@05NVMlMCEg!`c!7uv)V1ptXOpDTso)9BW8dnm~ z0QDeGUO2YmNVwmQR67;Zdw2?!CQlEpgi88voLe}RaOjngJHyYIikKPW_pgpKg+4|P zO1{5H$BiD#KsWj29hBH>hvAU=Xftr3`vMV#Qbp2#TvHo!JpUz&w)E$@2~nq5ZIe9J zqT}!9zz`oQ@e4tHS>o-ydEd?Jiri5z@qRnnUk%7vM(SfUGV`-iN>8tpCACQdU)haDb~QptelSpT zj1?C?a)vKePAt`{_yT$`J%|zJ0rb#Aq{Xb**@@lO`xbfg7bPJy`c!QUzFAVeK%VOeo=5X^}x@6 zMN#nAsE52DJ{%Lu6!M44Lof$RAGR*Axf?xdD=k&hUF3&g7B9hr8zS=SjL7a(aHaF; zGIvSTRZvROHwylHlvR)-J&2LYK-1rRKk7xY7#aZ#{l`zL)wnUJaz9t@r`*)ntcDp` z>ut-tO1wR{31dSUXJTqFc4`kUi9mDI=>EXvB@0q9sUW!+wtk+dq{IEOKrtBJ)4n(* z#m|XRygYclIpSv$BFWBZNz{G;)67~UWyk^GnXVzR(Cjy9c%-96->u>+HKt^8qxRk2 z-O=$-E@y4>+i2#<8W#ty27)XdZH{6O|%hl8MqeAvpB)VUc39oc$ zF>s2V(i3s_&bP|sc1tP}`t+`66X7^1k9?}OW7i|N&HbEj4&L3EV~bE6T-IUNvw41f zj#RQE+HMWGkxmasolaqMJ!DHqyncy~W2?8@?;oKo<|^>Rz#>b;7PT+hVA%Fq|95Cu zn0;-9VdV132Lk3NcIZjEAoq~tUe16XFo}su(7CVw$ka?c`UMAc3sKtM0g8dqJI>=h2reTe&0_k z*(Jok-+I07+>q>O^ph0K9~&MxoViUw&DMrz(CYFJnaheFSBfz3=KNT@ZeKaq)4th_ z{@z2g5EX*uLPB-Tij_^_m5pgKKNlHgg@QL3y#jKdA-qT?- z)B^gR14{{!Uuw!1mahn3^$<#&J`yG7#=VuV9-<*yWgp8q{j*UW7f~?-BNioAaVr+k z_NvXH-O8rAt|`s&&M2pQcCu5Fco3bpBKV8VuTLu?n04|?7B%|U6j5$%m*o(7@fd2R zFrz|0e_UCHm8U>X61PwMG@gLm5ZlOV>hiEQbdRHwSptUYqWRs%(`_?>T%7@vUs#Ie znLTO;)hQvVO5c8reeeFh1DU+N6duiLl2K1cyk!}-lc~sA2l|A*FOVF0qU&-c-WEDmKyn4(>W%+EtBeR7oQDe z@2?sSG_2=*u@}JiBND-bbYTq71Zv;auI*&wAl5pOt{eulLW3w=xc~ps*tl z!HSZ(SGIoG&y;~rw~aXn=F<2iUQVRuN8&J>;{7+Yu zVY7_#8p8nHH`ORNfjSfR1`Eix@4(YYg?s>8P(U*MF!OGr}-yJ=wjYqP!{KSg2r$@yo)a3C?xCI!1HW+-}bh3ed)BoMjR}X>1LSI9?!uvua!A+3g zr<)FoEQ<;v|A)4>45(`B+CX7bn~+ZF?w0OGQo0*Nq(LO4rKAK&Y3VMNE@?qpy1S%1 zguC|EQ}27e?|%2k{dqX2=^)+lZz!7u-bK>6^!0rDafNtqa zUAUx|4!Fu0?vSvOv67tvzQ@GB%)&*INx*V943oI^ZECijN&y|jWVr@7sqy7Woms4k zRliKufB3Me9yJ*u&mLpk-Xk+h)h82C<>!mLIMx1Gv$6jbcvAA12+FZyXTx zW)~k0#~IvzF4O~Di7^j^s?vdQx5k|`HOP9DKJjbTf43KK20|P4HQ@ly8JmWqj2v+Q zQ_*c8o6ER)vAXeIQd*cKj%ajsrSa$`hE0Q~l9E#sSiY<4y?+eRXx&Nc>K+{slaZs| z8X`xKk(R-V)SsC|qyT$pTMG!Jjv`{#CF3zb+Se9|`q^Mhh;z8BpG(B#i?Ww>fgqcK zPa?iSsAUpngRdDiLA^DRqz(4Sv2oJAuxa=hPM(HuD1Wl%UQ0f-S8yF{r;QTM!?vI} zto29@jUgKgKn-|Pm1T!?#ZQqNVkZ|{KoO5oEO#ba7uBh+Pl?hspaqdM5D^@XR>bx|-8n zfS|~P(vtue1R9T+^7B{rR}Sb}FB)yc17K4$V6q9Y#*(#AiW-8Z$w(S&eJ+9sA(%LJ zDKesT3Xr_7Nu-y>#jz zQSA9u1M(_>GV1^SBp6xPoPZ=QZrk)iswHwL`2Ge8zWp^-}-b07Udv|JOnBBAX(c zg7x^|@^lN(&~HHC?M3X$-$dVC8Pexo5afKCo5Y)Ey7P3%PPBXDX~FkN0w+c9&}#oF z5bZV0{h=y!(H24d0sev3JLJU(a2-|<8357T&BO73jt_GAE6UiS10e*@=BPA%?k=sG zhobgAl%P`NZIE^JMwvpG!t-eO-)8xLcnbC0dZ7FC3$zF${Z6^3xk7 zQS#pDldpyX+97*a1&HZKm(5?%Vjh=09Om9We@9=on`gO&-8=#%c<2PHm^AMQE0HSc zIT}&+DxSwBr8V9p{&aP_i5cpD%xY}tRzFALQg%;x`IFxGtUc=o;RgULH~ANd-x&>) zr3~p$KhlupWvgN)uUF!S#>uO5`Q*aq{n_nIrh)#5FL2^D^Zg+t&Sc=9F`(WVxRRif z4=_~%2*f-zfAS1OCs}8L(vk2;8WSD^o{}3$#peefE#q*i9k4~LB6N?Bf5%Bwp7%Ro zWGW!cE^RVjBS_6}x(#Ksz`MzY$XBO%tC3YEn|H>=W>-tDg!$?w=OVf(l2aWq=t?hg z;{s-I-RAEB_?rP5C9oLH)zRdo63^QqLt^PAzHgJp3 zVdz^CT~`4!%f~oxObNZwk zmoW^opBnmb6NmhOo1UsiR#Ki+1Z4?6vB8N5e*kDf@yB%VG8<lPwI!Y^siv5lPSTtm6R>`YhUB(xz}im2+NUslY_3 zRR3yBgsiB#FMNN@tJG%a$Ma$|PUi?!u7?%g$FiZU|K^)Z$Y_}*fDZ+e@Ejt2bYvNO z(LArwE_svf(fEi5;9zIDG6%Bqc=Wg?L-vL2gbXvt^93MYSlP_wQq^>NxNO&PE!(HAn99QF`RZSwq|<&1eeFzRik}^W5uhy2J54M zv-*bfva{YX>87G1CM?~4^}lp0R#z&H?R=KwJPu-HR|kKRa}3PJVAZrVa=z2{ED3fw51y0dkXz+~e*hzx@mo+s8h%K3~zGK890P!Eem2P_ML=;k*QCsJ^^e$n~bR?6H zfP$|S*;F~kLfOEePR-RIgXij&3~8R@;)qLma#n=(h)++WJiWk2EfRAaei)qpJvb7$ z@k#(qo789Dux)LS*zJ>Oyo1cf9zBV+xBVl_W6fs0)2|1U(>EU5iYkjeHUM_%8wZ<) z-@6nDo*xx#8~H-T+rAv0b!n}2R(CiRKyQfn7CHp=NcDU%bV}uNgk0JFpVx)72>7SS*t3cZs4n zb7>~jGoF>q|C)k~0_?0eD(sfP&LVJNm<7|a^o4LRqd>S)8)!IY*nbt8rXTMKFNhq0 ztq`o#N&%{lveOdC+v>0qKY=^Z7X9>0Z&ysaCa6_#edr`9)Ov3?^ULd9?9{EpW!u0G z2YXc|zdeR?t}Fs0jp!p_cX#`H)f-2aRTxP?b-$$#HP0k!pBs)BeAwxkIkZ6eIP8%C z>P?jTLwM(+&r}Ykc?(1-R_1nxPW4`HiZ=wjw_kDmj64?{QV}G6s!C4r$W7s6ul1Ao zrq9B)8)i_*^=3XKPVpo1-n0?elc~tSI^@bCE@k}0t{{{6DQP#4O8%0k(5CvS`DegE zWRIZ|`3#@6Kd>VH;LAeXUdjNmcfr5gyS_mLO^oLzp*Z8kJg-uNo z$nDzu5~Zv`F4VUYQu@Y=XgH#raW6u>$ZcmG_;4s8S=25UG9arL3H;!war!?k9W=dG zv@WVMfq+{1V&S5OlG3`p9BvX$>>QQfC|MnJ+OxsB#)vQCAnu2y1#w@x*^qrdU*~fQ zG=H}m_s_>@&{}1pWqIDAm5>qKU83;Hbp8lfmH*WIcwzH+)@cH@2r+Su^MFwrAVzUP z%^?^tE>HTdcIYVn3sNC}*sPlyBr{&8fE+R7q?Kp=;i!0&42W68FbZX~BA^;9n#!q7 zBZ;cmGA0gXayBb@OLf-?LM5H35tUNw%~X zH~n*mYeKiC_pJP@aR$;k-kTT|PF>Uyo~Nlo)xalOZ_o19y^nSTNE%KM4Sl8em1Pug)8CWl908ftIWqp;tlE&I;b7V*n9w*fR72KKm(shh>KG6 zN+r`%MEXYGsffn4(#PNTADq=^9W{?%70BJMe62_j=+bWiRxG14#bvWo=|bknrMd3J ze*C!$RC%Wbr!NrH<){6LYv~RSwtuI*RAfqnXQMl%9MH9SJbo-^#S{KG1*BX57 z5RlA3cv3cF84ms*-Vb}OhS5EcSwD;{sLoDP3@N4ZHERA-G$Pw9tXIxp0h8i+hmH1EE zC6Q5gW#UdBK*!CsT`w?Zvh&(OG-zljtz-A#^Wp9hbv7$GLSS~D2ssvdf9h<#yc?mmr{lj-xp{Wsyj{J9MzL+ham1B>O(ShLT0paA=H~>Jek>1=DIz`PoISw z3;*f0dcMrnX0CzNAia}vW*xZ)i@-&9@g684HNgek?gKwD<$?`p8l-GrT4z{<(ZJq3 z`)!awxMNbR0lD?EeZt#eBQ5O3T6t6`vC-L%t=6S`+svNk#_SO%+UlfO_r%fYK-M6n)! z%P3xY#DoyT8u~}}mpG{I+zQFmmHg_w>kLQcMEBkfv9|iGlwMtg=^fm^c-!FuFO+0s zo>^PF*-Z)2p0=KxCIA;&F~8>w{fJ?RACYMP+AT5=G={7l2|qWb)zKQprIi*gAc2* zQBBAuLB&eGUvod{5c`{&H*?^VV}I{KEc|IxtCISl&qG{IMi2*i;m2UZsq4YJ21YY68ua~$+BwG?oB!wLw>%n$lA zBGxhm-K@Lq`2DR)W~VNWH*+*3^1WZ+oXz<=P6j@@yp-rkPO zD7K#4z7mk6Bma4D(b3Q0jka!``&duF&rh50UYYg}Kl@X6ZsnokSjG>b@wNl~d|xG5 zbUO?merzB~y~t&v@DQSL7k@Ry+ZJD`>pW;YyY6yD&iK4M6Nj1|VRTfoGLEI0BX2BaVCDH%cg)w5On_d{N+y ziP$Z9%YA{4PzgOsK^Sf^5arjOLL0j{1wV^rG8yd6aR@u z2a#~rAPNmFOX(91Ll2B(lZqp=Q6St%~I*{&hn>n8h?TeO4SLx?tw_1^Z!T~93zN@q*+ZQ(o5JfV-rMAMmPf{U zFxBgrf2Pc4+pKsOD1e_Psk4~5j1?LmbDV^EiSoL#zOfHdp1J)hI2aor9Y2s7@AD`@ zUPvvG=jY(wk?NACkQ90u9;!4=AQ(fM@4IOWMgVNb@KiDssD}9OnLxy^T8X)UulbTx zl=_)H{9<3by`7x^Vi7v1`V~l^{%H{a&UgLzXT8d0EyZ^#SAf3Y=qYA-4CFk4>>7NN z3MRN^Ghe?~m#tYspij8pBVyi^aR1Uc@WbT`_pXw8#Muq^%^InvwU)u@)jE!>(+@db zeZ_LX&-m&zwYlEweV$lM@)DGnc&vTz!`SmS!6bNBKL!#?p#r5U|GqgH>!pw*;{D45 z%z(J{0l!*73>^rP_Dev)*S?5^lPm)YmX~v%Vx0kmKJ8l{k7v249j;@AT@-MZT0Zf_ zUCiX_v`h9z8p8N_nCZP3`XJ=k#uFMvwZ(t-eFxZ(4g$WPK>W1JrPrJY9zdZ5tYtst zUogR_0n=FMl7Q!iHo)G#7~6%E3%@#kOZf~i`sM<>$0+oQf=RM(5r zTfZE^oqe=kz0-rsq1{gPJ5b24+9O=z>#|d8&;CQ;68p~mpEZjLS(BF>3cBCxUb}ha zgOlmp=&t{`PN2JF@b24TnCY_tcTL^z2(Un2LIg+%{lrP{U!6{EiQ0^MZk?TYxLXmc zrZ?b7Ax;=r#E&D*yWBo|;DW+u35i>Px=0iX19UTJ(FYLz{^3Cx>sODdYad=GIHxM@ znvj#s%*IGyrcGXU^b8K8M8v15uXg~+z&g6-i4o-zH0m-MpmY{l7v5^vGNT+2tPt{k z@!j+zE4fQsB!S+c3Xnb}X0Z3XO_a)`Di4w|=U_IACHzFoT=N?)r)^YrJcR}PiTZK- zI8t;<<@RvMfv;srw{i;NS`A&UL)nMd6sm+O4k?1KLV&#^kk=Oq7_N`BaHCPO4T=Rf zq)4rJ^?44nV&cDkxHI)?S0cSewAF*BmYh2g_NU88V$6PKe^miA5?twvwot~Han_aA z6+S!;--?gnBt~BviF3CU_Zjs19b->W6gWviHu9&xHnNo+!0V6dG3^t8(t~6$p8f`F z6^Qj(Fp$iyeAKE}ZPRTs!F5VI z5U|lXN_0y*s^t373jjL3Bb4+cEQ>8!eevnx`$*L>;DPjxFQ)#r45<((Yz%(k_BO{y zt$5b&c{4ZsFY^E6W12E~6}kK3lXF3Sn)}x6$i!O)38Vd(?gKF|gJWtd2YAS} z08}{t9qWdak7(;>0}K5B_T1Li=1E>DzFu~g52`kTi29MEiJ^Rp15s1w*>d;x&Z*%k zlz`HMNX140P~t;!nTSUsWflUEI*HAg#EK`#5yzb0!V&i(-r=e43g3wT+O*nH762?h zURSItRUzQ<4M=XQw@*2V&ldA$tWS@H%2Y+}j&C+e{Wfn?iU^N->47T9wa6*le^~=a zNgX^DkLe#;0D08y9dfywY|T~-H$*GyUx&1xSCkc@A%TK5H;!}@QpNu85yeF;`;SK+ znU-=2#y2ADP4_*tC9@gH9yd+o^kVuOvJk)|>0kJgJX)Y7g1)@9HHneN^pP8)^$yex zlpyX4EH2ol4nU@LS;8k}9Zmk~CFbF0qYwv~uz%*~Q`W=Nr`% zJQz)%T=1?A0Ty)s@^NeCE5EMDqWOS?%EO^rXq;1@c!DADnc)P|-GI4*&{=jYK6vw` zMD7$p7JwRk`5ZgYq;a~;b)wiN0Mn44CRL2`Q=;t`pC|DljJ6>m>$#!U_hH-ax6RB! zLpVVtHS;gpm>bc^b$Jr-Ra-`dt?y60S zHX`u34!nXmP|mUSL9-D0ZXjk~!mQ7`nwyvXjO2!@(A4~Wwq@~=+xd`Uv);i8M7er< z{#IWH9|UP(APz;^7rY@zyR*!6$Twl9Z=t;ie$;44Nbpy_XWz(cxBV}=?YT@b$W<|7 z$Ha|qhj!@kS&Tp4rPMfC$|s`+nrbDly5cydUWiZo7Y1eki7lWI1MUo4`$5p|iiM|#|x zeL#*dvK<`hKzrE!U@$V7)A!U530@50+MVIS%xnMkdYJ2Rgaf|wA+6iO;f)EWh>e=t zVGdPvUDdGgUiD`KNMGKnCi5Iv55lO)%tMqJtvvr66oBz{4YqOnP*`3gVBzsrzjyD8 zJhD(B{~jacnY0W?JW2e7j|KPC-!NL+GZh=&EF&g{q8BNK9^A2YKuHcSk->rX+P~@6 zh>+rbT`cB7_9ALEv%$=6hRJu~)A2mkN!7H^OP{{IHpzhjF2kJYahv*RphnY5%TpX+ zQxY1(9}1mhRng-dnt;z_pFJl(LP3jZ<6W_v3-D5$wkAj`Zb&;;Oz_GI^^%1 zfP zY@h770&G}c{b*h-W`TQUmn>t$0 zq~F7M*D3DC459_jzs%!GTE8*Y(0z_)p1Ua(o2D-^oYbhOL+W8A*zUD4*Y<_(&OKG> zxk8xEK)_&Df{DL}Sz9Rhg$-@AO zhHN9wtYQ_cLoCP>a&*4a$pL|vEnNqYe9;Lu$B&f38#l;AslyvW91=6l{Pr)3?{ius zQ3;()8x~+MevtY&14xuQ%SR2OKhdTG0qo0H4k-IlVFj=+llxVi4#FI$Q+e_oFNo7e zy?sPXd7mH&!#2Sy!OYm>7TN7UG){tRGDx3(5r_`sc+~B{=eXAERoP{-b|Bc&j$u9` zGboP|c#A@Z^1(3~EKm*agb!s8@igGE<(4p)v`uw@FR~30wI%t-iW58W9M89Pqa?boIwFS;bgDMz6Ie%G2Tw58=sRpy?4)KA0w297qaCrr@WT=m_r! z-`>8zypUbuT*6)w0LsE(`EdxrhA4*c-c*UxgqLpKN*jCud2LplW>sQNT1wA^(bM3C zkeh( zcDg{xl|BZL4aE}*7R%CmI8joJYc+AQXF>8FsF$oxG?j=`WS3k{QUqM9jhx7omK_LH z(Q>fUE-rpatdy@{F}i@-4o=29PoyloWP*FPiyvU{5ha&8(1PF?VtV238fqvg3ACKB zo6kEIg{-#f!V?4ND4xLEVL3rj>zbP4CaM@*5JLOPRjpvYLP5P zzriZnh(U>JTy+lSh~2H={1S`HrZ)~qeEc$M{~<2{96d{$x*A}XCxCShFnad| z6@CE7j1mu>9cC5=HK5UdHNXVa^HM^tm^G*}hABcaFUY$I1XSnF%Xuj?2UpbQm)fRH;jZLx-Tj3zKL$Vf$4XkkA2|P%R+I zhDK8i%JoA^Ii}a2=$w-s&tXH*rD`SV&1GdX)%b8()%35{jR6zGx)4 z?g8@CC9YQ@mA)*~Y97$-Gf$SO?rKF`TP~IUxweXzEKsX2{Hpx)E zjPe1m$?oe)OpsK?8sJr^JR|Mp{=kQ87Suf}AlE3fdm$$4AY6C-q_J$2U4C~%qZ1z|>(#I( z8XLQSdr%;#NLtDcpe%=uK7XM3f9s&-hV>It*K2#b@kLXy=;HkjU+gSVjOEjsmm9nR zC3IE_MsRQ|`L?YSIor=&j5K$1f6;=U>O}19~WXNW%)hJ3xX4&1!zBtDxI|hic}t z>$y5Ao=aw$rzo3)la+G`x0(RlPeuQDzdmBD=IT)KB3to=77+wP^exe~A7#WQ)<2-J zs!x~n9z;@R_v_*Ra<+D;n5-idH|;#P7Fk4F4HrcFi3(hvLjVLKq^K0s+6~ZDD^k34 z3~-zQOaS+YdQbB7zk553M^(m@&m14pOZ&g*PF!tU^C+kHPl6E;Rgy=W_ENk2DJc#{ ztix(eTUo5)U$j11IlJRsssfdzL8%mLyKcloxUU!ac-bQFR}o( zorbB&(!B2C#1Zg4Z~E28T0I9ALedqFhM8Vmv0+lXgj@&Kb<_%fLV0{7Nx&r~k26BO zncXi{Ork}X-m*@qO=fJ}g^?@S4t~sq;6XSA7ehTHJ-N`9QYSGo}rqTfAl^qhw(%jdLr?r?XE3L3aOZ7balT^7Or5M{J6+trC#v18)RKGGe#m{;N z8t_cP{+*#n{?@C0CnwQpL?m%D6)z|oNQ<<2XdyZjR~Mk9 zX%DCB48x@0^1V=GVid)4M2cA*lCDTgV#COtXBX%&0>^|lcvGx~N2w1`ZdMYo75%uB zv*Fk=LZ%$Xy-}S|RNdV+%&J7>jYN*fh)#{8t5Vm~&|Q2NbEjAvM6<@DWxxc*Jt0^5 zld^C|&;kX8XSilyAO0A1K!yq$W7kysNZzrFOij#=mLA|(3R&(%q8$D$V%cG03<|cZ zkJ)&=7;z&lpZEGUhcW@m=i>lW&2rJcm-pgrauK3};8!X-%XRuPs!UszX z)3GXwNX72Mt=%E+u9bY?riRR%A%f_T(c(xTf8O3+srvr5`}wTuRr&PTzua zv4|~|*OO=mn;>d5I<}MRVUXp0E@S$fgn(uUlQg)+sHb~YJFjA*dEpy>Y6zen%3S7pUtbb6cahG3}qC@~nbenu)HWEi~BpPKQ3#U!Z1fA6&PY!qL)&mH-SVz;{Bj0X9s*#miYT*e29 zJT$hF#STG{aXaMGDKxVIxubH$4k~YJBY{Ayn%k{T6pFfXLgC@!WU8eHkXyRu-fb`7 z%s{**wl8r=(k*Wd^_^TVt!KEmn59AC8r-qDv$%c?k|$n_1MaiR@*Ufn@mNsm@8VlpNl>D8d0$f^b zS3dH$H?QXH@JU^A*WXPoJa03xzl8}{n9#W30bDNC>!jtV)Tj0s^w&jU_p_iyn6F+-%8Su+6XtyTcqSyXgMzgrlOD)7Kg;Pck0`hhUk2C#80@fq}I0DF;)Wj<+vf3cB8ORj$$d&iwt8F{dD5q4w`eI*9 zzY9q(=={kV)QM>>Rnw;6$iD59r?U=8?_8!Xc&mRqb4c;mpCpz7El{ce3HfiO+5oEc z(d^$Q_L){1jCzHBiy5<8ftsYLF26`UEy#pN82a)`$;&`~h+4o($%_)Xo{~QLwl9^) zdJGeo*-7JnzD5uk^~6AsHmZf9>Qj^*a{dm9+ zqWy6t0Dp{Of^bzRwIil~k&kQCc*hWth_G8^u%4tYkT&Cz3rssWSA;{V1F4?uL5Zxm zKl}wGx-No9Vv3!J9{@2Bq^};(5Jy&2ol+GNnTzz;5xP$&=4)0jo@lgnUTzlnbNibe zEsS`XG95Ng2@CN4$iAE!sesE_4M&+4YA~52QGED}6ifo+jp#wWAmzn>$SNMKpCGGG zaFK1zZffJ@v?ULn55ELtg-E{su|mKUbHjnFxTxL!Q8F}l0)m!axnK3PxtqzpZ5_@Q z#e3$*k`1=~@7E_m>^+;AZjRH8zB=)ReQbU-r@HhT8)cR0-^ zTqwvC0*q^0l05-?)aNFL(~KfF2op>Fj6#F~9X11TqbGrYie0opq(!b5x&Gc#oQd9- zJoTQi(v<*Ps+e{Ge(it{;T&NB;SOX+npxuel22q{Tk=B=Of0P2iAr*)BK@gsXsRL3Sy)?u$_&Ioz2Zrdk*n~uxu55Czl|fHCH@@Cm z-GpJ1%5^t0t!-7}i7$s$mj`u4w=tQ%^zPK1)Kkz3(bNE&aZ=tHo^C_N!34jfIqXt5+dGe{0*J-LJ}N01HPc-WF!(?N4RNzI}_r64@44tzJ1*J3b;SZ-Cm?u zK$b$B>Nf==O_!?xbs`KnfHnHyIRa*(N=v#i=Q4KKFwH!wVQwhFHE$8Y`U5CmAsvye zned@C5M28d-JbYXam5&0QeXZo*l;xd1jCs&tbEAXVcpANLJpb|l$k zv=-<5aj?5_50~R zGKe1v{T!Y9+uM`<-qgOl#cI%>9YcQRjb>!~N0wjSTy(7t!Mrt>C%*i~jYxhoF{G+8 z<65jB>sSl!jc$4$k={2yY;f%8b(L_hfjxiUV+SjtM_kvi$Fij+TBvdGK-$4laFIHUiLG-(r}soeWd&#q zV{3(drbM-5De2P2eQE(J8<|Tyq?3~ll5ylOwcnfxl93DUGC;Gkl%7-CeAZqVNKI7T zTj5q76l`*bX$uWO{#F-2b)qN^3xWS2pe+RIrUS$YDL}6!eYqw49o;@5aU6+Hv=U@# zo1F1`IY{dZ;0VO%JOoOP5NV9fYP`TO&8<3j8)<(-5|J%<7D`Jv(+2`Iw3ZU5{^ZhH zq&Z+B>1YdNfz%j#^Ir1c-o)9JJxo;pDsg5vyT`9Itq$Fm4#cS||Jj5H{%+1HwJQ~E zSu@eaHwguMy5*?RqsbbaO+9t;shalewy@{O6rB>Oo2LKfQ#HX)k8D>(u+lbGC_bVq z3Tt|Q2_PQA#8=0{=3<)B=1S+L_FmGabJdMvR|Gt_Yb7*TDPRASov`#pW`KcSx0*aq zpq&N)t8vnqG$b?G0=t@sjDuNIUk>^y~>_RHt;m5d)-+Bpr zI-=aEkBuQk<`0tjX3kB9)QxqBAV{{dNVR6|ycnt~W1yn#Lafa4hU|L}*SmUQTrEfN ze|HCx#W<4A{bCow`e);?@Vx|hgdc1|JEIdS6lWcJ<8wP1BoI>L{raC zwlrwBe&3(b1b~gj-xaB~zuOER&lUW=hs5uQuz|u9s7a-I3X!L1Ri((9>VJQB99B4D z)zfkRCP-~4DB2jgSxdLALv3CB!~eZ0kJl7=`?Exu1QbaHptlSOj+RysmW_Iy<}!-Rb8@}CDx{-7s-t5+xr)DCP4Cq#Um-HR zDSRsWm${2+A*+_KhW6xHzQ_h-USjtEp!3hZc0kvN zWPG4?Uumao%~aiWJ=FID%RhWZ**|VTJsYIn9ODKrm}00v-Iq@$A26xgDbE{GOfg~K zfIKN}!1AB1aC$0oLN76xc-X>=F{MR^Xxrw$yE5zs#_ zDh>zN=yZ3YNc3c;+9xOLN3}7^*{sY!89i3cbQ--^CuNV0KtNY|-9-(pw72O@-N%;` z9|0ZQ)hNJAi3F{4P2&DC~-LF7-oCP#SNX|2W<`Km!N;dG3Tbf#nSv)HnB$W*V)5bx?2TqO&F$eA{4p7^dD)|RwuV#i{is1HruoxjTsL=@`eS*`UGlDut zrxCfU7GYJz%Kjj5$U@m^%RjbIiAlFkfJJ{#Il0n;N;k-{>E|GOzH6$Q2o_7byHuFm z2E247hczuvusMw@q57sM^H)!59 zA)%yZS$ySJm4atZ_a=C|;?7ST=llpaUmy^!!tyTU1L2zsCybe#2OE+(Z>ce-hWSAV zx{+QihcvBgFT+K$IEY9OUzK(hh_g69eNbG5H?!pY?6!{^ag_sm@`Hq(XWtla$s_L# zR^|1|XN9hGxG(L0m~_;jW_U?TyuRb2?}>(qj_7iDG`;|cmzL&q14qE;lDMwsv-%9W zc2T22B(f`K{9^MOk}61t5!*iBX3z8&0}3)T{i<9wki*vE6`jrIrQyC{NT zDJkHAPsieLcBBG@Tl#z8N=Gk!fbgZZUBE#W&ZWCq}XLoD#| zhrt6ltIKC$a*)5kLgq1#;JcYbvB?WLc4A?2qF_tON(PuGsBOo~Nt$@jU&2zFO17at z^L+zT=xRrbw!|9Oofi!mK1C4pwOMtp3EQ_+fijofPlWrh1C+q{KW1cM(qX<^7G;i> z{+j3~QBW6*FKC!ItGwQITpSDdeiPGEV9WF(0+z0pXx`>+c1-}5;vw-79TM15p;8xR zi2`PE4F<Acg|o2Ma<15gve7BpR>RWMpe_&fV0h|01Me+ zVt%0_{+{YECZzMm6sx=n%hAPOJRd%y6?6&Hwnkz~#rAsz954wHOvzzdQG1Qt%dgSe zlwjsuS!5vNjIo8-WHQXTe1@P+9_BS{lkHN*S|x`+jn?sN$g@t;L;FsS;Yv=`e@%-^ zd`UtMEtBwx!M@=cjQ?Aj19frDu1LUY@9BU|m;Z)uF+)g48MczZUjT9cJqkXF?cJnZ ztc7wM*1Mq&B?^tet_7Go^eok6voJ^KvQoxqz_-93+QN_{u6-8LNk{R&VgdRf>U@Ll zO|Hj)fs{s777~SYSe;$i)G$@Gl@L&Kyr^_(I#@G9-}}o|ZaF(|Z&|r}rYWBz$e(+9 zw_eeTAhz9dz+o*yPq`!oJ@^O8k{C=f1$VFH2lGe%sxaOJxn@0AvBh`vCLG6PcMJHo4+a4|salydiXG800FZ50o)o zWelknr-%zgi2EjtmgSCr&8S>M1~SaT?REMmm;DV!I(~?xo68O@t@Q!kR3l)3vWESVLl`jZ4Q1b#P}Llz)Qv|t z9qx(wvYT?43hK{*@#TP#{Sr7`KvYyE=6``!{x}b%V?z$64y|pki13|lus;S!*d^Y8 zkKs5k$Emyl1Vdn9Qjh~%_5K3hi+0-O)EdSDjnP@{^wWvYQoGoQbCbiXa#Qjk(<-BP z!{>^yb6BS2Xv(x@iP6>pxlvE+fNz}=AudR}F4TtuAI`uuw%bbC7G$<*cEh;XdZJr2 zk(k~Zzbnc}NyqW({0Eu-ZhE#j2EwV6(Rf=mxH8hOqNwP~V{7PE#C@Y96v%af2;_rwLSlD@9GO2J9UE95cxLkwm0xpj+V=^|HLHyC$xDl0 znQtxGD61@i=;q}n(1a%M%L&X&*E`h_cC}79?^>m6Q~<6arZ9Dv9oIjKdFe3W43* zfH$ho=v%(>64N1x>q=#;DpxXS)o0X;nEqns!N~Q{?9WUvz76RSK4F2l*(|>?Gn!W0 zlQJIaa^*fQ1f^QM{K?7hw^p+d7 zqtjEGy~A;OUnT?L0r{K*NSrfw?&{)W`mTd>dw;7c$0Or=fe%Ms_~96H&%gMlZ!}3* zkk=q^eElew9QdFMJ0rPGCXv|e%hz`pO_oOiV;IFhS>_7--hI5!6kunFJcLbr?EJN^ zyJ%8VOdF1HzbSYFj%jh`C%L~h2s=>!fRsf*C2P8BWUdU{}1>%}7oCjKPr3 zm&Vvm2tk)J+6PQDsM$Zht4xTPithuNDKsi@xj!JFPai!R}pbBVpJt}QDETsh|s zZuo1!LJ@2efEW-D`P2B^VWOSNiMOONj=Nwor?FXgjA8>o2wkXr)g3?GGtlLh(>&kv zlOG+Vz0!N@x<^jZ$Op&YmxXaUo<6(iQeHsO?rVb3Cc^aYgm2bjv626TT+itSX|wYr z{XQ)2gmjnlt(+|rm9C}xqt(sgL71A0xg5F$T0ANe>c_j|b0)saGblIv_Uu>sR4hup z1@(ic+hanwxCG>m5=Nz;HykbdWMDCD?UjkSEZbVudGRITfb|VqR-qTbA<$P3@+Sw; z0lN;@5VqsBlI<3wvbyHE1;hcXfd>RON<2BVrc;Txz6z zxbNkhj{^AB#28A4qc7wib8Q$LNH6#LcI)n-1muGc?>1%`*Jl_jV=YQex5s zC2L12^;i?NB3aXFC1-Rgjv` zZFCm_3N}4m@7YqioChg+ycnFg*+Ob#JD%CIo~XL8#l$c5jC$k@x}8IN_JnPZt=`%^W|HO)c{fp#F?Vw9 zI5A-~w%g|joV>iu5HDONl!yp+^#_u{HZ;K|s)>wSVrXQ-9x02c9fo2Cfe5qf+2N>C zEsTj&@JbMWlY=?;bJtMm|6%JbqpFO$w$V*VZ0QC80qF(-=?3ZUPU-FjX%IvZknWJ~ z?pEo}O?P*5ZuEK2`#s+|KOHj0UTe+1t~uw;Y~;UbupQ8+TXTMhO%By?9C@8~9qTr@K_9QJ$0*c9@0^RI>PHGiU!?k0_s&2>&RfC3uaLxbVdo`yo%wA- zUL;KK-@WkM^>!C=V-&KNpSfTuq473#J~gS&IhW+6qn%(FKnwUL!kaKhgtp^na6<&d zB_+|Jh0srI5m#F;bI7AlIg@&^32vHyK~@224f|nXE{c7qPh{p|&-qGhZ|xw|#&7#% z98~PapCu?)8Gw!$RkI#={b3{_CXI*`vfxLF{!squ>C(3kj*uLsJpxl>@nh6Y7;3q} zmB!0-Ua!mG)1zKV_V%-@BT#0KdqfXIS)J#TgnLU*OsZ(rV>Er z8Ef~EV?V99iTgU05XxjvQ;D>T1&M)x8IY${qWhw>l8V3UL_j?yE& zJ#vvf+|9cY>I~Lg_Q=g%z$wr1^R~Yv*KmgrG}?77)>|EQlh)T)=%=3C*h~4%aUOk6 z-{gJGF@vf;dBQKVy{t)Ykt(9%1PN!k6n}Yu?}=Yo;{@pn@L4gMsmI|R_VQzUTVao>gelzIsbrw~Z}xgz zeV^yBVX3G4so-p+D$L7Jml1?HB{1XgSQ~3n%#$Ob)LT|LWNq!Nw@)KSUN2?i@%B9TV zdWA|nfH~sW^d1R^{rY-`uy%?rFDH>TvQ!drt?xO{sb`nz&1qMzZc!Wrb}^JQWbo_| zo#dB0lakJg2ts6v`?svc`Ej2Ru%flyQeWY^3AY>&Z5Fc1C_v(4-=0**W%l~+tK#q z^eye%%fT(-2CL!*JyiE5L2n~@m^|)VQ9(!qsoRUNBPD+RY>AbO+qAYsCVD8x320lj z8XM>zb}a?C^Rm~FJlFC**gNmTxrWujUem+A|J=adCXz>$yQTu8(*5OJ16iDsbOtU}#l z7|J)7J#3Lz+gxoW44uDnb_LpbauCL1*S!_~c0|pc^7o%&8!t!Q(r1iSled z+Ny20yMrPWW= zLDexMw$5xLqg(wFgf<%d#z~WMTZKTl0JibKJ)V))#8jL%+s>BaRBB!h;y{wlz!}GO zQYV%Q=149~w^Kt#-x!u|sU=apqt9fJzo#4QSUlJT!nlV~h>elunr4{YnFvRN0=d(dts zkGuH%#QdvQHrz#&w`D(ea!$u5ZlE<8~dJuexd3udA>P z*ZXkT3sJr^%DRnXE?mLo((v$awvN#b+o*1G>qT#dY!Fw&Rj~SQF~Wnpn!R)yI9a>) z$>E&l-N4)LC~=)1jlHm_Z(L)?u#O)3f-%pfD|)~A%D~sPgR@lo=7)`CVc17_8&&3&KMX1@97npzwdASsTPGj0?2h=sN)-k?Dytqf@2ChFTQ&wJydM&% zj;osu1j@s$w`x|9a8j9X@~QZ{$oMzDqdFd?klQi7@)h{7d(MCcE?>t%7T2ga=?oqF zxXf)k%ScY^9+X_KkmM8`b(7AT94MK5Lv(_}|Hq-TkNi*rFRwFH_?ky5D|kWg#x+uM zy6~X=&|9uHCJ;{M25Z~zcg!0YSpsdW6c+3gp+huoaNZToCtQ}}cS@v5@ z{v;Z-Hby4GK&hs8^KUP<2`;*1!Z@2$9j1r0Onp}ML83(^=w$@(F)y}V%sCpl7j=B) z$#n+0M=EXu-nJ`+x=#MAEa@CJ0W~GF(N}GZrK`mp@qOM%KTZ-@F^^&NnnN5U+|(4> zStn4!x~LfKA_80hG0$d${o z8JlISVsMn41Gx7gGHoO053ZO)AGS-2HoZ1?x?!^%Y~)1@LjCy}e@sTP`yEXC1lHDK z=`Zo+ofMQEjjVmf`b?%t?z%LZlWeS?)`ecbR8|@Gez|F{yB2o*xMjhs8;Ed+IH}gm zlI@ypZlr@VFXr;d6CKIIGyGc~QM4*DZSJ~z0#mI$@;T}wKB7a~CNX z%4%BN-lwEoZ$Iq}e;&4K?Qdb9qB>^x&NQ)&7DWeNgm;0ii8$38GLqt`7Ua}sdJ04z z!#+@Djb@16r&ja%j{6>=eE@gBZXYuCwljd;S+j`7$kdQU zpWARtb>TWzm^B}C3nSf5jr@?Pc)(0!9VZL|V`@>I#RVcft}QI(W4Jfj;+Yv*FT9+r-mVqL2MZGs|X1 z|I~iF%FdjIxs%URAK8o)q=7<9;`4Gv)n4ze5L4@M`$824WXJ@9sim}v~`N3yW~frjR3uz084p>0~*AA4s+mSx4Y z6lzw8O>Hq(*W|W-enm}gpn^RM1CZTa{DZA6+F^JZ=HG|neX-btws^C}Z?li5w7lwX z^84>$)!0w;7wUW=Pswvfb)`|YOY;QDTvi;tNa7{IZPbPPK(>;^MulKQs!@{CVBBv+ zSJeVxr<-Tg)J>@Hh$}_>^nR@Dfv)Z-fhIf}D^yWY4E+ry!5kt-{p6~ESvXJag zJeBzwt44C*^p7119{id1By1!are$X$ihBTp`4t7g@nI#tuf5-<-QqUvQGC$po$zU3 z|M_Py&M37y+p+73N?2Q&4O?{#>zhVRaMcJyt0i&Vp_Bzwzz?=hF%zRQAsK99+IwGN zUZJF>gHh~AHAZ*$?!EM`M|lXJ-rt>)R5NHay!N#%aGuH{w*^C}91gnnX!> z%xN_mlLX*(1~{Q#H=7Y-Q!X20ux{ydjfP9zB$6xr`C~{ev(j%vCY@&Dn_8T-n4sNc zQub>#0DC4_5ak8aeqM1XTeIVe8+6o$kJDv^f9(|2=!6}UaGg*l=VRM`=HhIz8T5o} zP9&X6oScotNZ)G7c<%V|RLEajD7$WpO|a~{smLqTwMXK+ZweJRt93TNIi|2}=G%Na z11O&Bg9@6o1r2a>X5Ky7SfBzU(#o|b+NOM}8 zN#8gUZ!Jn@Hi#;%9Sn~WzX~idp>gpTRCTgN9yhb9XLs5v;bdoik9%)H^I$ldDdg7J zHj>1$?M}_7Li73yWVtISZS;=3wA4Z*|xptI^}JSy-;VLc2cgI+;}WgDDeNg z@7|*0JngXJ)90)u(_y9D5{?uiZE3Gvp*{88(|*T%rI%Z~)F61_Qi%;!0{vDCypymo z!EPl>v%fZ8YzDDW<7u}#MNl@~=XClWsAQ>YBkSTJl#oy;agF62%|a=^|Rr11yaDB<<76#3cnkSF^X#AXVfMQt>-3_gBCwn zoBGzVn zif*@u+{L7aKCXXz+n*~=SM0Y?Z03iNJ2q`X7wV)A%l#U84Cm^k^E13Wc(S5jqdE|7 zswY{J&(DI6S4sa9s)bu8YdYNl4m$lfJFSuWKT<-A2g>cq=I}B8>4n1bfW|N!W8Jd2 z*Db>nSkV>atQ~3_TL|=#6J7nfs&I&x`$>;lXL86F`tm#q=ELi#VtiBe5w4C*BR#br z>{ipgb$w|bbGM>t(^t#85j7uaHAXXj-T!9t{}Rhll2K3eh%J((AzJa3N9Es-|DzZH zZZ7hLRU?RrK!kaMEiIScQd(W9R&?+15EN20R=vN!4P#6LKTfdjG=;#Rs~WOQPVOL-tP!sT|#Li*m zk*{e9!8kQyW5RT)1gdxHhvLA}?EWwUv;^A!F_-^PG){m`47}Ml)1H5C*dZciwS{R> zCF*$)la1LY^4%J{81wOtZ?e@l#(`Cqa+{v_LRj<{d{5bo>b~Ojk;=}LzK~6;2)$-O zt0Be3&vJwEr(c}-rIXz@<;iS|*TJPXfj}fcnNS$1U1O|ku$os2eZy?Oa)Cqm(caK0 zuonwNzE)n8s^J?@HQ9i9d~dj)$X~i~37~)f4B{^p1^fd69^i_EnNip=UtO2ZTzYpd zm?-wzC^|3JhFx$|tM%(OqScP*!+T8`>GK(64`&-c!&)`#rnJbF>jvDx*=0ZFSgd3` z#zU)CSe>e}wKl$f)LGu!8&2i&N)2hNn{uJ|N1+M!LneN2t^vv?kbO_3@qP{k)4{j7 z?djH*2&v2}?q*=(tq<0Jc&;EhH-fXMy+CrA;H|_Df)ZAwDNLRddLZG>ZpN8O% zHWVtMM}U``Fi)|W;&)?r&YMeCqA8(mwX~gyPqh-m9ZBNy;?6|171Kqkp+mY0dK<%K zQ)#f8j-kX-xs4#b%B^v?Ie#}Wr2Pmn@?((+SVK4dzf`OoV60;|SIhC*pB4Q*E4er) z)fTdEZU_nG?+Se`hvQ=?YF!uM#Uyl)T933RN4${6Hc5ICtSyTV6)I+!v96wF*J<9> zmM?v|YxFx8k|NP{y=+PJ))}=d{)I?7gjyQd^b#W;h>&t~={0(=TJD>_tIgRkhK?r+ z7W+Y8Jpw?^5K4~-3)#P*_P?j20kld+GOtK)>ITYfZWNdOQ#v|oZ-G4QxyFtQBx?(@ zZcu*n!8E3%tTl_XG^A|y3@tKS{{bED{)u)v4sbU51Ku8fJH0p2`66|A>AVJE~V}GD6MhlX^`k)R9m~Nkb(ML)8!0K;x`}7tk;^)5QGD_%X#yc=_|H7WX#;P z%I$%n{`BI_YZo9JauJKUbt_+De} z3A957;dv)I%Ab8u5KZac4=6YkP}aEGEh zZVu&q*0!Zx0uol5%s0=X`K)!P=HGYaSBmHu+)@0Dg>iAm^&e=LVF(pKc48{?iF zx+UCO_CYJOTk~-gJyv$@Z9(-B89W7>4X65`WA#~-^7Od%Hh%;t?#7+$c&&o0<1=y{ z6V-|pe=FoXtnr7#Kc{x#4NU)Sp??rQ0$Y0uY9{AD7(Qb%loIGQdPxhh7!?E_Jhr=i zyZRMe_6%hcERg>W=q z7SVOi6gNI0!FEhFu1@w2?UVbDy2+EjwUc#8HKWEmCk1}omo)MJ@cA7{fc^Cq29|4Z zWj&jhval>WcJfnY?+pAVd~SSdEl}V*N8gbZcy%0v8o5ZR^uhT(8PDO2y{Zp`rG6v! zC5b?LFm|53@2;v7)h%;Iwd$uoNtKCJWOkGhKTlj~V+~F(XY|)=An)I(^Be?Wb^zeS z^BixO{p8NrhxI77)5Chm+N84Fav&yLIMwqLYK_$l3Y}U>$7W(ocsTL0u)}JCwp}m( z{Q=?qC%fkE&vB#qND~P?P{~(9+w&RH_Hy4aJXx?xap9ttqi+nb16hZu zpeOZN0#1(8d{Wgk2E;<=_J(q*%?3{9IEC#$=#q&uD?RQYM z@h+x=s?~NJwJF7yGQCJwOu+qzI3X)D1?k^v?yncUn7Cc`7O8FBne4@0iHoK(AKaMP z$fn()-CZ0;=Fhv>v|casMz!AUnb|19-rfs51=!T>2B{XS;nyzZ=jP7wD>zG%edOhJ zKgp!57vP}MaE2aD-RQS@BY7_Pj)E>BDFemoZ5P(>2yMdjwG4 zcQDWo zMnETHF}yUfAz}>j%WoxmGyb<12okJ-T||g1r|8O-tPqmWP}I+xwH zgXJHy@??JhV?{Zm25uqfpbc5k`i6Wi|*?>g5zO?gJk)6;mU#vdGP zy3MXO=Lb#`a&V*FHL)fDX?~|n;)$o*^nL@;W9_t`#bM2-Qot+D+4GS1Y$s>dG*Pdx z>}OZlx%hkA1q0hizH93bnLW*q-Cr^do{-87gjUysFy*2sqD(x_hd2q~z_#}dw+Y&4 z;i4UHuyOU=q4*zGK5Ou$D~%f$5Vy^nVTon)Cpnt5=8Hx~fpP9CZGQjc_+Dm$f&s$G ztj5VRq;WA$QI!7nWK{D`q&lP1;%6DOT6~mVlL)K!+*yn-zNtmJ&QnsUgLsv1F0lOR znU*qvBnwMVQsRElI-x{|KylrGsyMNfWm3z&}`;4F6Q&aC&|ru<$$4I&p{(~~*s1kjzQxw=bJe79kH;^p_1+r(-K za;75hn|yYU6a5mSWb2NEkbO+&c1$Hj7OOWi0<{!aJtfD_kz1eO9Hc&0ou2vHo)Hx# zFP?LiUNQ@~e*S5-roY*fCLdH)FB3n9LG@EQi;h#^{U18m$u`eVEk$&>+G2N0&fWW*Bng-cJG0X04!ZX={`7coeq#5hd~E_|L!_f zKs5aPM40!DX|~H#e zyEIy~)%OIv@N1u+AXjjAUdK`2pWD`Y%e6Nv^FL+yxI!Rb-t_Ju=~MhLKxca6F80|Z zFqkuf?1%DcVDAVjAJ&Ynn@^4c29rh)NhbjlM|^xdn`RD4#G z0SQ}eO?3KiP>Cb|+Nu9H(wL(InX*<8LsBGd-}N(`e_{#-y}Jo@3<|O_4ev=kHSbAF zl2tftr018MIszWI>bBrY5?`NWejm@!u52SDmbPrVSvhVvu(-yaMSsOf(ubk7xA%pn z2IC>`XAMk*p9^8hQFDLPOzSylZ`q@wGRsqYJrsG$>+1HrAOjZS>B{i0MqK7i?HSE; zM!*4@y6xi($7?V5XV&v@^+id=z`+`4!z8{pq|5-76e?C#w=40Ji=IQ|XpZ1pl|iM0 z(QRjNLeucHzv(WbjX2|D`Ra`n`+q0VZ(##p`k^a$eKfsJ8u%qp4z;gF4(n*mWlRhY zD{$vixf)OX#gL$|V7pXW#<2axFO72y{3h_e$EXtS(=BV%?CzkC(90y()5tuso|^Xk z(UjK4&=pz`LcobiYPj7>Iq5nd>0jL3B@ziG4ix(Q%5Xe4{7XU52^;13hGOCMvXi{A zZ6uNFB6^I#lV~P7&L7EDK8kmm3o^@dDkLo)?`8VBi>j>$R)jVmoTLK9HaE^vvK-P) zHI3(^9`;B{P(G^ue=bA?T0C>FsPjLZ%3;?4628LU|-1ALPBQ z2jt7+XH3tDn7+CRMa9}ftn*^~4humchZ4j))%$r}!J4wlL9t>h0S`aTDI4;9pIZ&? zhk=A4{OV5`FxvI*M1(;=UxdKrq=5akt{c0Nz5G*{c+Q9Qi-`)?!+A>dC8y)oSvSa| zQGb3@LC`ztCj54ajejKt$3~iqlhVl~`b1sIK8!q?h2&TQ_lAD}OhNs}ib9XfPsh9G z#cjM5Zi5hMq`fEkwCkbdgb$=M|3~um*FMmw0n>o)-5r^Y$MhmmZgW}Pf!Z|fcQ4On zR&YMZK%*fUy&1Ve)hBCOaS?65h4oxQ+Sx;XBCs;Fnabw_YVWXv3V1Rf)Zkj13~}kP zJSTM-U#du zm+^7Nlztuul?@MPbOdqwZRf)a4Qpp1%EB#FF3it8*oiTnYrT=i8I3@oyC>wd?TLpc z>6wrhsQr<`OM6+jaNIG>bNuGst=s?|!YFW^uE2VgEU;Rb6Kz?u{Uv{erG)oOP;ZUR zv`rZ*!)o#0x^{)D4K8vod{e?49HV1p#e;$BQk@duaHl~$=sXoYi8Qxn1E~EAoB4Qy1)KU{r5~En$_>mm&5iswtABufyQl=SV~K!EX?(ycoq}W@#7Rpq zc-1+OCuAh#mHofy6B_wwIb=|*sm%Fp{QThb&*E|F2_)=nlhv*wY1qV< zKks;fsJuI*B&LeGJ(rrD64x-V7m8}MTYOZf0l?t-6`a=-#!u@M=l918`DHhVmn7}} zcOLS>>eh{yE9Z8_?N1lYd+pIlb@qXF>!fW4eNp&cLYk##Nz-V?*P8_O!FXUCEIJT7 zT-}rD;H@-7_;XVNzGT08eQwo%P8(p4Ia@o!1;bvl9$$s}YBE}D{|J${>~mLv)H-tB z#koGcckUDgH~5-5+)eoHr@unU@-omUkF00>lImj8e&xlv88KzUDeB4vdT6KJ1r`5h z6x{IZSIMshk8#fyyn4?6^*h5ZwZ4t@Zl!So=ZU@4*L3!#x%v~T zYMn1z_mGs6FM@_;L#%1wzW7cSfJv6UU-8qpbgl$`8pUv7c5dc zt@|gSFe854&Rlz`*5FixZKymh83>jeq`=cw@$R6e)o3#jl{Q!VdCWm=rbX0QP7~2V zTcr>y@*z>pPq({*UVe%h1sd70U(^Fu!ZI;0C;^^=`zw$>- zitp|@a)wReWc6^`)#mZ|vPK9YFhtPZp*@-kybxF8wsYR8H&El(_t(a{bP8WR)fr57 zdkqeO2$Z&2 z$Fr&ZnBPIKVNO!JV~nRSJme&Yq~S=v>OwbP;rk-K@3SiPC@)ZnSDI32@O(-{U5foi z+Nc4HLQG2m7<ffe6(Vx6Z;IQ)@lLQN=Y11wYxI@Jh>i*TZ*qh)-3EV{jF->rHi} z9&*g5PIrLwV16@UK~eN_@drx5OKPIp4Nk;2gUJS@UValE+9tiO5};$Ko!uIKm<|PE z7@`VJQxLzyd?Rc}8t4edp5K2PDizf6j?ak+1SS#+`01ex-kkvPt`A3y zIK<-6LCa9~2p~o?9TIq`q%>%lI}U-tOW(V+Ksre#)AsebhCtNdl&ud`?2On!f*gn! zyDnGe(wc7XY1F7M8^@F24e%c)e=!ED@z5ABAwL|KaLH3qYnc~Dz8fb?A*Zte2aI9> zLVPE}b{Vw%Oj}U_S#7R8o^JbsjZyw*Q9~WSmk`XsP+ogkOrEYikSJMN1`O^zk=m$L z6eerf z^n?BC@Q4fP8|qcAI=@rScXRgv-rkp=3_GDcy>W!$^ouuhd8Ts!G;iMmOXb2Fyh9 zOFb)ID=+J~(va`R&bzO^&8dyjD%k1ZJP9Eyq~;WCTU5n5pAFndl2yAO|0qjoKncDs zf|WxKy$hA5$%&+pEd9pHbv-^VvO2Ekjc|CkW^rp3GWhg`eZr~#)e~QBHq$Lj1Me6Yi%cOjZ|`ryk1cOZTI|eB!7D|5+!Kx z>JbU##uba(a`X&3;VnE)JtPtvu8Xb&`GQUu{0l%Q2zn7nSoFhiqr@Qjrd2{&H|Y@d ziWT8!Baku38)U%Z9}3m7hCl!$@~=Vdu03$#x{C-My1dM67g@t~DB?<^+dx06A`uVI zg7gP`du^opqKb#PA_!tf9f#DbqK$t_k#T!@H6Rd>)-W1guzC?(a3CVf<% zb9|E`wDZ&O_Qo|ls22q+-dS-RCG`Vgg+)F{iigTEczuBRqpe3HuAuTDv{(i68^NLk zZCKYsKJV=5g_lw0Mx7 zP@9|4IaF~`-5f&%;`bomaAmEF>+TMPPRDyyR!WV5Wr$8|Z`4En2f@yOU}2c4PfyH_ zERSrTD*a-~RVq+$C#2Kh9NqdZ?aVGhVJ(NN=X^Jh0Tdq%+QQUHI-6mrdd*R+hQv`;P}ZU%?jP8T{(_Zl7RFKCYFjP@JSh~j^I5->w27IyR*K6;^u z7lUR^I@4BQp)EHb;%ak4Mq9Iaeq8MLaImnaWZtSIv4@I(`e(dcWZya>`U}3(W>G(p z-+@XVR!(_dP?Z04ljD55{s*?KzaC84W_`TsJn^X>znzWjc<=i`5Zz zx`zeFDYMp#p^kJV->TfyEiZ|81#{NN)LdGV8WM?%Fmu9@6ED6&_(O`$ga>qd={&Ig z40ATfaTW~t(Q+RLq9uACYnGQ_Lfa22pj9W3&7$~s_Wfr!L(jT8=Fu!-~z#+Yk zQnPnbYydyyqY*;>lg+_sE2V445oBiwPeH+LZ(@ff+dCK)YW;{+?F>b`1^SR z$J^id>GHQc;e?XbyLfL#!Q6}J{aNkYZw0`%m3D0-mfx_{bynK_R3&EKlbjU?n7H)r z@$GivdGZ<6*A>>6sZaW_D(f!4loF+&<;bIyBoQ{iK&%$#7)M3kclsI9rcYDY$@e>Z zsh)AsC`BJn=gCucsfkOPwq%w}et)dHfZOy9iy;nfi@O9mCs(f7BzWwCJd1ilLOm+X1}I=#{goITv2Sq0 zfj%wO1FrR7?7{GGV8*6v7rED2zoWhy=+^UzQ{JG{&-(VDTeNV|4}=#)G7AS5d^!oj ztbaa&NN}|^aD8Zy{cLC4_$4G3J&HdI)U%e}wThBUN6@$yLudWwwLz0|_>7}f7@dZD z{ek`dYd9p^_EkwL-lrQpiJcF_b6s5!%&p%BlJO}bu#p{YYDU!bfIILR#ws;l7+s6- zEG_X_9XoD+p5a#3m+u>l9r3L95wh~xz?sI%AJ)IIFGikYH+^>raHQrh`i)>0@{)cM zEI5{^O!=Qxna9|4QE1U?sa8LI-c*ItvPxIE-bSCmVL5A5P5j-bb30q7+l&J6p8m~5 zO{&+NGu)g`!>PRxzxJ~w;L(2#ZkE%o1bkDJZoDxs%hiAJ%8jo2x0i`vzpYS%Rij%1 zs^kt?q+K7vkzgjs#<`6^FzD$=q^CH4%fUv$3IKnB0pQVK47-C|&RMJ_4d`6vE1Q&U z+PzDPFzDocQ%1cao-~TObz`t0k^RI*OU92Hw%^SyL2XcjD4qQ-WqDn_wLjF8Tzd^J zG9WsmW|vq%iXV&Ky$UghI+-?HP5`~L>$qNGFEm{cqSyWWY6rH!x`+yw-ZMIbPp#e$vQutgH8rDqa0x($}BOPJj)zHY~(uZ-Cs0)jXLGp&DMKy!yH?ov&6 z{j*Nb`2}(WzAr48^0K9)F=TDoD24ccdI8|;O)5$kQW;2fQk&6nYJT$)+KPtJ(ZW(_ zxRax|EO204LN1H@pazer&W#lW4X|E;>ys<=9vmu98#xw@wHk+lB}%FkGFQSW!h5h1 z`Cc%|q|GSs<3_@pbQz`nFsX+j!wsp2L12WZCZfE?xl7t$X1SYWM(doXK~qtRkHv3Q zBnsTE+olK%q zPM9=efUXynO;*C+>8Lb%uma!2kf|`p&>ahn#c78x>P$|Me$1oyxi@U+q#wP$8|x z)e?ES?w>?84|PUDr{27#Rk0%etwM5jTwGiyuZDPf^un?-z00-V<;+#fuO_+bA)XVM zUjlpkygXf^vBcULiE4ZJrYU~BjXgMzZ^FOOTgp7lo8)Rtg1t#d)5c!IRBv)_iL`Mu zJkp(HjR@#JYLeV{hl~@~zpGGqw-Pk;O#OjbYZXyGsACR94A;?u>*ri%`c00u*ftmT z@TBJ(YFa7s15dq$il-c+;SqwJ5+7yTdV6I=6uzra?G^Dq4>WLY9Y&5Qcw8;i(F)azIi}a+?26l-ER<`ir{^_~i^{FF^&b}em^?{G z`y2;m^Ah!%K1X1+7H7H9)6)LrdKk?pr(tB|kc{~%()tHk-OAFk&F}J(v#zYPlp&-m z8+zx_u5mNr@(X|?KJvu`moiZkwO$|;NDsn>n`9J-QtrT4|9kciX|U9C_NT3^LS?Te zTxo2Sb@M(7@(d!y5SgZwwDW74c8@gLeCEIpU)jn#v6Wx#%FtE}WK4r<$fmgOvUyvC zMHy!#fVb3DN~#njxQvG2xr{ikxBNQDSp{K(1jtex&Hogns?zuc{#qQEI3D0Q)32v_=vG5TDlcqNvT@fg8$ zd)B0L*H6djA0dW`i-nCnMdZ6P;)gXxR$VXZ;&O>v-g>&l0dU92`y^i??VitvDiI#U z|4uz*#0#eY`OFe$$-m1_{>~)%1NaQ3?F}Q+l?+f0N`p-|18C( z`XtHmybrL9L{7cMAe41|Fc^FY^9uE@&bU8XQ#veJ9;Vvta2^|Ic-d1^SO3Iy{U{As z6ef8uVH=!nS;VhFzKk07bUPl;K}hPzKYtYNG>U2_v>;vN8^uSmjcdp;Pxjp^)h3T-?W?>4c zsr$JnQo}sxm9t83ww>|shFMP;t4;a~ghxyk6Pel#JheJ- zYK)z+^;8s!V0UpCffze=C_9y@)m+ox#2m|x*nkA9zSjL85WO1eeZom#2&$YJgExKeI)9pHmnkVZwE0G zOFhaYgul%K#Zn3XqkzSka3TM!4}f=|0AXms1uS`HxFjL(lfg~hPzqXUAT<@gN@AWJ zLR>$z%Y!zvD*P_hD+jWytC>s>S^gF1bBzloF(oskc>(TFU(C#}q+a6_yEVRIb9Ys9 zr&?>~drT2lKY~l}R>2KaMH?`;yI6J6f0PjOtNmcM^y2z=y9?RAQ-(~km;G)MOQ^6I zap?4dXUg7q-tYY7=7W|d=|sj&3)d{w6#YiashLO|rlS_islB0$7_7GGu!p;7?Y@y{ zcwk|QlKdb2n^FAIyWy|HK!fUAAcfIFIGDG(`9g5FSA&PzQ5Z-d;5Q~>48u%8`63J= zpy7RESc84DXgAU@-OTNqkelowev;**>eEesCC|doCrQwK|8-qSNr`1Yk(*9q_tjDf zkQs8=nwmeVXf1cU>`(jEdf%R9AFDr&=4EHICfu->{i)n1qgwRXzB$IM#n9>`+i7hQ zTClqPI{5t`5jgtqP_+cfk)IWC;Mi@SvO~+jc}{EbfW_~=gu$i(v*EI$4m9W1KbOoK z3||O2%MSZ~?a@dTVb)B`VN4FCsYy*a?=BWK6&4m|8~2ymEJVA8i61I#HY*^tNHoZX zA?oD(u~>Wh@m!XfZU2!kMmrfttRq52HBJ)R7yp&W!Un@4Q@e6t){Ln zU)n}jR~^puxyIGTWFr5|;dGg9|Hp`+2hb#xdzr393i^8Am-X;KL}_Zc$gQirdDp|1 zyx;ma61q#TC`EU{{&U(qP}EkmlY>ibZ&OFTc1Y=)b6n@Y_;Yyt^4VK~&Hq=m362P{QV!69AO{$P~j{ zyeD?;TaT+mb0Q85^A? z6imq%?3FVe^&7aFNj+EiCnf(di zAd|4#>Bb(F=1g6i(TG=4)!P5Y+l{ebrev8m-PFD1^GV6ERYRy1NxR($LlJ$XjH{Cj ze>>jJ+hbHZMB`su(FRejkyUA}0BwNoWe|jO5o?A^A`Q_W z5$gE&*9Ksn3R07v{l;0xcoK{8Rk)>9n35W5>4lvC(JMl5O@z(oq4w;ltAsq7HTuv# zBn=Im%EN2h*2`&Br2ru1W4qWGQlYI+(4hl>bF>rL6>AV?YN!wro;MoUiL1tH8%mkm zb=IYzf}7@CLqkIR0R!6Ng(USnIeU*XEGvVRWb43ZPTDo8=Q$9rO8ONV@HD{6m({CS z>A$pa0Hy)Qt{2n+%8HxZx8V*#$+5TF!#uasHvNZwEcG?c&6r9xiFLn$Jx+vLEcCEl zFU{H)(Zib24No=X2Du{t1e&PlXrzl!<1d&4AU#DoiP3t$+EZ{Qm_Ei`y^Ej5s-671 zjAnx<;2esX?s^xbhoE?T4Sk{x%7DIkENcyk0T%Fxyysqf89#U_&+m?5wy1BU33^>7 zT+~J{$K~a;(4hGNCT8q4vj^Mex$U>TznW!$_-5$5F|DgK@gS#xaIDjJQ0DUTE%GrQ_bVqS6TcmN|(K#BYSEb|e3LSN}#fj}v= z+)#s;o+@F3D!OQ{1d;FT9$^M}NcmZm8D-C%dG7G*5B-As}~_^kxJc4|gi5 z#9+aj7aA*#CA;D29Vh~~C}KZ9uvcVwan?Gq%8W0EQMkM5p0CNWFB!D@g8rZ7!IyUiTBITV9v9`#t^sXy z`Y2|}yssDoAqEbG)u;`LChDlWx}>&$Lra&Wzvdh+6wrybk8jUHSq6hQrlO5u7#pDY z^(C~N02n#Ay(h4gB&dNgPz+dRH|>}t$N?L<3(ALfS0TP|yctHH`K!5er1m&KBgku6 zr}Hms-2Sw{#D$FZ9{|Kadk4roFn1TChPKv(0DO)2H+E^fpi>|m#OTs8AHr};mIl3I z1o~{D4jZZFsV-R${4NkcD3~!jbV1`DlNLD*%R1^N#L~xtkuEQxT@=uH97FI$wo311 zLP>S_|f4-{maq=Y^L zXiYy}1TN{@uacdCtIYyX4MHyzR7(R8-hmKq1MLT0QCmI4JdkcpTE_XQ#SVh*qcX{l zDX9BZ4%VbtL87D|hd$}dRPd_jWqVfR%UZCLC`>;;k9YO@FLybK%5wi^3jG6%J^^~= zw&xts;j=>N^S^evNCmj(XV4D5y?Pzm3``x<*1tD?S;5tWn6l9Rs)K_84LBq7Cwg{m zt^gTC950v~9)Etl^eT$n?-1Sbe{S9r(O!F?2*m%o;s*sN`ZQu6Y92@V?=Q%_EEmI! zADYi>cp~$6tls-lD_U`aDD$&k5BdiN{V;q5s{i>KEO_pI!Sk45_XltV;kQe$#IPq& z{^3yU@d`Sq19eG%_1h!@kj4)(pxL@U_+h>P(7RxK?V|7SFAL8eraTH$4@_yj{LWWU z;{NR`8<#MMCl7gM%%sEs4m5|u_U}M1ppSCy(7Z)>gDtHFbK*p@iS0mT{vg8Owjt$L zn$9&F(4Xulx&PWLh62k>+#v+zk@o+0969N~2KBNo&1>khu;teX% z1*t&nPm6K$F9(0g_mbZ!9K1}gRt68N^s?FPt16&LCfLt7UzBwU-X2k%=Ewd& zjJ;)8m0i?7D1GSe?mBb|($XN^-Q6kO-JOC6NP9@>E&)LlNeKx-Lb}7b5Bm82XXd@G znXl)=x%Xc2Te;W50?bh>qJF15CUhnp!V`NK-Jd_hbpK4X3w_35iX_+*)B0*eAk7o7 z1vF4ebl_yJ`wc2C#C#7h2S%}~#TgbTB4Dw0^R5;!3H zpjg3mq2C;ThQEzNvyrPi*Z6bF+x&0*fYEXR0ujEJFMttWG`aY1IASaYRhBIe#TCQo#Y1{3h5~!Q7|ruUVC-V*$yID97@7=#3*Lz65+Sj^C<&8JRvks^NSw>H{TRh*>Up!)A zZ9&NNKpu5Cl@l>A_R1b#07DQLh4x9HYwNSyA_0ZU_nE^PY1P}$NVF*%e`Y6`@&Al2 zcLM4UJa~yoSA`}@d!*2LnSY220C2R{SLJ(^_fF58lh}vy6awvJzak&h<%)<&E4o8x zHcCggfTSLdxvOEiP) zC%1cm7(e)0m9(s04>ff|2a|Gh4Stqs2khwglv>~+oWA%elBfnf%D7P7{@4PCb(Msj z>?4YzJbvrqkw4Zwi^HTy6sF9=HiY+C>PTB1w$v6m_lIhu&j9tag!>$Zi>tg%`zY{- zQCP_nP_rk}2diKGWihGnwDj6lUKz@v5Udzl(=&**92=DD);;<|5f6>-;079s#e#Xw}lZU9v~2RxX*^prR3trA$aC|Tf6F*3?TJ?QYi_K zdfFlV=db!@sYavQshlE{HDQxk7|?uC^(ixOC;1OIg@N~gPU!bXB6!?U-Kkpx2!N)! zWB{TXwD&j{lQ0<;8VwO3pf2Fxd@=$<&Ge`Ztc2(cf*tFbMT7OXZeapKnQn}(H&Gb+ zsXH&Oy*DcpUVs0X8vGTiB00B(CU>*I6NCT@b_#^&#cB{aQx~R?pC6kxw&OS7I*S{f z+Aw^P7syoKb0lSKCKujD4}PTpO-TSBI>G|E2O)pf-a)Fg@9#BPT6E@nnQ?MVACh3U z(__8huKGCRr`@RMG=B94dhm&%HzA<^?C7_>lA?APua8BZV#miE+r)XP(9xp+Wnbig#%Hw^VA_Qv=OmwFLh|e~xmT!bpwbAF-Q|{^SNg19OQ;MooqdpA;364Qi;cVfAAw~>asleuq{c& z3E*T7D#7#|p`(6>pGBZ-uQ6gp;mb;O`?A&PPx?NIMzmp(CZ73U-|AvZOn2$;X=rd; zy4Hv3zhnER>%CR5gjS`950IW4DlQNzC#xiJRD-l3XVkt(;Jxtx7P?HUA$A{);f90@ zfGk0KjC*dMF4CGAi%>&*p}BSzT_(+*F39@w;q2FsH5JqEO|Cwdp!*s2&@{PFB0KV~ zqkA11SHE~El*fPj)VX!H{Ne2qKgE=;K*r7Iq$=D+BOd>t;l+$7fc0`Re~v(ZD!f-T zZZ5DVmC^vm9mP!RSc?ditI->%T&c`%6d%w#r~tuKC=}c@Ay2BR>`O*G+aK-)SceZg z3e^$j985V-+l{3bLK9;wq;R?3_;k$n(awPtdV}m|k*tOLmRO zwIK)|m_Cb2878f{H?WvoD9~tr*X!`)_NG^cI+nW3)eLy=arjZwzwPr20(jI-3xXTN zHp;=NPI|0ODczJ7NinL1dGvcMdN0*sPUez;T0rTpWL6IE*53y;P#`&R==lQ?GbjPT z1W|-|$u&|Zy<~@4OcY1FG*kkWEMvaQCZ*+xe-iTN8Ym)udizm9 zF=;|f7CCro-P!=g1<)S;Z+9pPlMY=cz}kEIWEqw;vNUQ27y4+J)~@+kHE^m|=&hbQ zAUcNs&TVX_zh75Z5k4z^%a7f}4ACEak;LbO-J)ugk!+Du4g-8g*;Ipb512Y!OX=B# z*UprU5_*PZiwD%7GR2Iz{eFIUWt|oXlNR{vd|qYw?2I>7D}2ZW|9!+KrRU5PE_!KC zWr2`!aM5%EmXZcQAwhL6CU|v#Zg)7PwVJn?7ke7mU?%niKW}894PDmo`t*S7;?se} zgYiZGTn?`Tz4h-0oB6;F_0H?Bhl9@<&P6HcZei(e4T19Vuo8ScCx}3y(yJvg9cnjZOr=;KB|I~v zsEtX_Kw?p0R(n=V#j5^xU|48B;552k)Uihkd!@(Lr-?Obtui^d|LARZKU~jl?3asQ z5lBC3;*jyS3x#*0+_t-y}Z5# z9M}XfE4~sr+(bN4L^^+=G;Y3HWWb#HC`=z=h6T0x{Jh=6t1D4|=~wXjhRGxkFYQ;T zB8$PiiRP0M2y3#Mx4ft-YnbDl+tv-K@ z=q4=x_1Lvwl$|7Nm~>umx#{d5D{CA^_=VY1njIfD7n;k!01=@0)}QURjaXjHf`T0w z!iGvR`Qlehj2BoTE4mlIXkXp$HumnvQUf@sP7k%~^KvOIX!#Sy%~|Z;PUvJJRxsK% z@A|q1#w-j(4MR&9>_^~%{&lrZQTO+{U0v;&>zN#L8^>R~`);cI3Qn$QI=S2h_7`gL zq<-B#YazGP?WuJ$>`AI`?-I_uTEUQvyKbF#`blJcSm)5a6+kw?n-~cMbg=nhdH}rJ z1w!4AQcibwyV=J8(PMH4b)D!~LqM;??)sbQ@t$!1`iSy*%;FURN`Qt7oC2$@0rNb{ z7-ADT0{N}S{^}L+Z}Pm~+pDNxt@iG*2j6T1F+oN)@+q@7h!zBafqmu8)dI6V@cULF zi@D2M3bkiTPPJZh^L#)YDb08uDc+6@Otgl#V_m*~u@Iz4*kN+7nJDt>3Vs`NH_jk= z0a;dA4-1!krNh?8lZCbSYOW{I38BqK7xV5%thDde`;PVDx$dFkd!euod~YX>5|bYw z!a^B%YX|hC=1;*_ht6ia!>rrqL&nQv0q0zlgdJ?;QYNN6x=$%bIF)??y~mx$n*gx> zuUi?Jny%RR0>D`t5E?;G5PW=IB&$Pr;cqGBmtIbq$Jun5C3w~ax8F+O)a$27-YE-N zqkMJYsPVn*@bcnewLd!`>~69RrngIAjt7XS4E zJdSrCB!IfzjXJeu@kRn1LuOqq{lvGP{U2R|2ywyRI&lxd%Q3W+1(bWb+YOQd?jI;x zzc=IO2v$q5dgHq4Px6ZZ@i9wF}fKFjqwBRPIKE9o`z zl;g~Y5iTWM8zAif&yUdfFT#*B+uZzI11-g%he=B(-VdD?Iqtl{1J^bK&S>l*5;hpe zL!L$ViIx`Mw1tq-?}vQ+jmza1m0^2)kZc4BWzu*{&#%qyPM1J8}cASJr^{QYrztGLyRRaAEs4 z*oo|6w!yd5f1^2a+__C4npTxK>)jP=+V{qG>Hn~=N?fWagir&xqSVZI0OZ>kph`Q+ z_~?wk(^n+uNC?{rp0X(xbh|tjF`v}=x*+&HHBz^wg~3~Y^x`1ibvQlVA{D2O==j(t zlPJJM>ZS;gQY1ew`*1I}&gj9*Hc~F z-f2X2vN$H*sCH61&lIV%2l*2~+j}7!FvP;UY3niK7?u2vL2j8oX~42GONUC|8enZ6 z4Q#);m!jVVWs#BB$jtE2W$XrZcbu>G1=zQVF? z?fvt5bIz#z-6?yaa@~uWB$^ac_yRU^)rz1P)S*9dvhSt<=uvXPuGN4R-OJk z#nE^VQJyerukYH2^v723Q~1F~$>T0hW1Vs+GsEa8U@wp!FajE=(EUbzup zN)HYYc)QEW0PLQ6F0M;7J`u{3XH5e6NT}SosPC`NMe?=x+s#j9>3XdM-;)JbdT>=D z7)G8Cc4_m=&%@4Zqk*%RCg*Qe_@K+bI6#8vvt-k!mE+TvKL^*(8am94kQ%;~)(PR~ zRg6;S3}!D2Hos1ORWQhZ7HWVglmUqW%?DMW=bIRA6>ergc4WtYmeI=s`>cwQ^EmQc z+nWtnWaA^4+U+JCaztHzES@qrN~62rHD;od>5FyA8&9%604IWvVX^pygh+siM-PsR zZq`>2gnbdtCz}wk>&~ku#1*)}Mt;!^)=?lGLnCFLz3De#!(2X|=s0PpmHmyCVnx+p zt!SfrhACU56aA5+?&O6#um;Le;j5M-d<_ z{fFI|Ck^>hoUta`tYzp{tl#1(0=hf}q_*^d1=^xcr2IV*nTI^;^x*X`*fZ%zz$QCv zQ+}JzIr%LMr8n_Qm_Rpnv(}*&aRx*`?*x2lL-Uu{5s|n`ETte#Ay`o)FHx@{ic5u;3tIhQ9u}($CxT2oV2U8O zX2)b!3!-gI!YvA)hzG@G6k=5S>5m`@j7lNLAY@ixW$-AX+VP15lts05g}fJ0XxjD| zO)zln0rL}?NJx}6rpCFnzo&mMb=YAy1Sem0I8jCNNUJ zbVW9EJ%jHL&VhN0v*e*(nlzZ4mp?PqmUnx)>8a0An{`H!kqa`L>nMhYq%c-$P$WhguI9 z;B18g1y<~SjlSwJKaopXwK+xK)EDC358gU`c58i#7|h}6lTwhz{W17{m+{_iAc#0= zXPvIa?71odZ(ZCdq1IBY`1~D?wlOm5fgOIlxJiU`H!1z`6@@vS zdf5Ew!4VU3#8(5Stc`PI^&{nm{+mfTlSV31ze0NqRgjfD+s zVPBWn3uA-wP5Q?!V;g57pPY3kp`zGu4lo2cC*M~h_A8@VeGA{|oruai5*svT2PYV* zEp#}o{{GtUS_;@lAt6%|g(^lAlJVNo!N%Sm5kv17lQ$!rd6)+GzFl{N;tSssje5x* zx=#vIzuRp=tG-M?6crYFH}bM?-c|8%&>%!iF*U)XG9*qFIYWu!w{g z8U$VCD0lUL9_9E)RZ{Rn;B+3BI4<$8sZ!t7kLN38FYFEJhU(6sTlZ|SkL0o^K>42Xm+#dPkr%+8G!rPHa75|BlCVD*j8)Cg z<#lVE<)Wy1o1;jhm!ocH!#L!&&J$EXwCA^TjFBs8$Dxqah(GX?<_& z<^@y%viVn=)qf#?YvcoG)!BYj)b)I4Mjdj0rzm{X?5rqf7^<{B?i{@M34hLfjJ_V7 zw*P&xo7VI03VnKT6DAWZuw5NFKIMoRI1Z=4F(6$k{r$Su7zNY3UqYO^ZP&lfE~bAo zJS@rxZ9D&xA7@@2{A%}WoVep(QW^BRCjrVKAgeoZ_bmq8_vzKo0y8x$G^yRHf~*Kf zb)BKZWb*v37jd2xZUZzcM!<}M)rG>>aMe;9*#O!E&u_x}-P5716BBpu^K1|U(eqVIw0)!5f;rB>m5JtU z<@q?06<3d?T(<6+5kdn&(qLRY;>aHF(wN!JQIDpIXQAy|jR`ClqI{Mb+j?@)w!MiGe+pk}bxk zHGvGe>L>1P%H7$kUkUf$d$44_6fKPnip{ob0M03Tad146xZHHHUZpwZWX9=YppI1T zBx{t(^&&!=d2##(75T zt)zu&`89`Uff2`!3hBkT>vKXzfcaq!f0)1c0x<8{kP`)6Ta{{ED5V#g*a#1WSu8%D z&wd`s$}BY5Q#?K~DF4u1JSg=4-?Y>M_@7j_9w+Vw-5)F{cxhWE#?jIB6@n%8ThM84 zzX6LH+)f#NAX>0W#&0iTmMYA#^ybC9GuC#1w*kLP1v4lUz)XapypD?h4Q}4hg9%_a zPC{#Vz$etD+ZzeM9!+Y4?Kq1T%8M;GgGmZir=*<&b&H9bo5mj>fyK7O-@1MZQ&3--&-fxz53i*A(4G7_&ywk1$IcXxY)m&a62F<@}#mFDBwHF0`6WEJ3pa4 z>2Nvo`j40?8^>?vtUAH}{GqYtrHa)s=%;V{_@eg(Usjtni_P2I-msDGstcP|E{aET z5!V#Xa&`eTKM5w+2r_iv2|Ip!k?Z-xIDy!@mb@e)l4pgb0+c}#8ukmYV|V=d-TOB~ zjd+Vq=!&6!dh=)332gBpFVycmY$FYsFtoKXP#vwUa4ly-+goR*dfA7+;L)PN@+T5iP36M|u$pi}Ifv z#-vtVt;YdCnA(#B560v+xN*yHJ)3&cgJ-GSKhkhX2?KSIQEgmlm1`7q|D$s6w#Xr` zt_Rx0TXxy>GTblO%b(2-#~WH+bI`HANCs#;6sl1Y@{h&}GrRV)sIY<91lMSS%T%Fh zoDJ9eS$B6X-Gbgrc_jY0vyK*!B_E$rqEl8tE%`J`W3`auRh?e<$BA=c9le~oo!}7l z!iO6PTF@1CSZ_1E?NC(g2{`h1v*~l_5!$F#UgBIi(2zU~5VcSwl9A5E`0sKR72O%F zrhw-O`QQ~5$Zu7K6m}zGGgj^44t;oMXj*r__j760}jtAu%{(sTP zA9x!TXggn?TlTf`^Yz-eyDZ|0uK=I>TSdyi580FMpWZm>V$^ElPzC~bz!66v`R6BB z^ipINr$Fyep9yhA6!<(?6%pwnTv1G;3Wg6@IR(7mk={SvZ@mL1smFxz65pNc#1FXP z>`7FyER%q@_aCk~>T>zE>W1)nECvyXoumjeaF{nviSy;2(1HM8?{Cl(Mxd1`EoCZV zrWPXomXz)jsZ7fgo%NTofaD$E*qJ-MYMz2utiqlz_emC-6LrXc`aTy>x_b$zIPmmb z&3vT{h;u*SOCL5Cs$g%PlIXy^)>T83vEA(6p1V8l@*+=G>9Uw^!_JE>w09bq=aV-i zJGo9sR8-pfB|Cq&&T>fiHbT11r^xaaa6Kaq-f{FyuF2%-8nCbUXb$J?)oyJ-MKBil zrW(zE%-i_GypM&v-sM~coI!)VUxl5tIvjpJ`&qQLy=oiyJ$G)lwrTb*Xx9iecv{S9 z%d5G?>}9hKcGo$em?ca2PApIYT(ap*Cl>c*GD1V6d1v55;iPuvUF$gq(Sa|&Wi=vc z=F|F}juP=yKh(VBY5G{Oqyz}j0~Pcfcsk znrQo|sfxXW2`hT|a;`5gzDLxnZJxDl0t%WI-%*K{I=^UpaU@f6K^(31pB=l8hYh*B zcXTl;4iy6&PEXelFl?c4i&6AXL>?FbIozOhdg)8V-_tr4OnH?H2VZwXrdb(7qiP7) zm>8_XEY!;tE~l!j=e8w`Uu}4mTyEr3oi64KOJmRfH%LR1xe2UTgVwqopPvYh+o< zQXqwJ#qjbfZObXY$!lJXAfJ1fcL}ek$nUl!S~1avdEbFF3-)||3BcXNK#0}4DbzO? zr9|sK-)eljjSu4jVBL50u|)ijBA)va=@;{xY3(FD43a#g6~>UxgNiPW;FDSNg;Pt$ zlR+V$lN@1)5`T^tfiR?FmDz&*X}#ODPbTDOrld*+dxsRJ=(2N4B1gr@>I>2jC6arQ z@HP<9r=LyoqytqMp?NHgmDeH`6K%jl-u70< zC&|5PP6HruHkV8Qp0^U82XtaWJ@M`MnfKiR7iXO`Yv_O^bV{_NIZ-OP&qX8Y+KH? zlvlqpnC0Hj#IF@laxy$w1`WhXdTqv@3z(2~I4LJ~Av?5i%^!3inI&9j;t;6u*6wP; z`#py$xChn`4IMqc`)^VsuadR`YRDJA9y>lDkA|ipTKXc5Ta#Xy>N;>WroGVKhEWaeH8T=BPA6c2Iw#jjQXM0&BXnn5V|py zTMldXg13~`6eZRT=Q_`tuu{;)OKO;eFp&v&sbhmeL#Fzgrta1?bU=dZ)PBpQ_5S7G zVJ)sQnp-3~;~34%^m`vrEo`7SJM=9k#N+^wdm4l5s2@-+#P6CBsECgc!6IE32wv+_ zwHj$YA^Z>5m)3Lw?Pj{Qw6u_j27+|eiT8t1(0^22k6#k1vdsoKt=3YoXsX1`*d{pj zo;(*y_oK7x2#px}XhRy+yqw<&wU~uxR{_I9#G0==L4WAZa0?t4%^c|iE`BCt~Km8ND zP+(DXt(e!(n?l?M%I6K|HGU!|g*yHH20U7wCG&KEJ!R>BXBToT4hRV)1L0Qu%GifWC>=(aZx@dJa8*;MfvBAFGqK~SN2#CZP80lpibt= zViOr{oU*xAe+gMf@{M5>=&Ex&5E{3$3|PkwZt=h@gg*B^7G%Xit!iR~XMjoU^vb9)-oUNqc$7=PIAthh;-ZukUP zp^)6(j^3%%!pXB3Tv$M7-QubX0v}1oYNj^XB^7vx3n*+;_#jN^BZ<~Ok_xR!9NKVPlB|<&f}yUbViz9Srp%S^c|k}K3j8N(<>$r zWUdh426o)JG9OinTcfUJhkP4YeWLxO2Lluj7g7x;X(4*Vw}xyAPo)@MO4WLI`=vKf z!YJLEvac#B5$otWLCg1^3JXeifNv_om03hIKs=2`N*~HT#8BU2*m{{{r_o$G>f|aj zaK>ZQR+@6|7Ngf|8?my*6+kDPi>CXuh+p z{SL7R;Ws5*GhG{Iw~WGiAt?!3IEoh^c@qMD`VumpmU15w46O5b*j!uffIa9fS@UVL z77&lYOik4z^}CXVvZ}}D!^|IJ1M4eDS=vrVM=9wi`j~~-o|R8>sh6+uz$EwoR4T&X zR$4clz)Wz5n)RN%`EpBTMIQ`Igu;$_l<+x;LdAKg;ZqEo@ChSaRt!00pkB)v%%KG* zp@U+iLWIN1p7y|X6LoHaB!bx>0uT*Vs&n2o%{6ROSP19UrB$z5$fhi{prtjir)azn zoPZ1TskI%~hxDe9FuX^Kh@4oz@)l1Rb;QVheVpA}x0Al%0@H_$OEf-<4j^6w0=0G0 zzue+rjX;I)Yb-e&GinWWOw?JAI*d^D*K?7asX9##uUW6v)$C*)arO%2ErJ9q_`Pnf zu;KQQ=Csh?ry+)D#(9T-AhO`m1rJ*$)GYALTK~o%wopQJsO7TUbeX04Z6%t*x3KEb z;9&4YYl_W*_+l(>mb9FoeL6It?A$)_EysWU;g2I|mUbvUkkvOgGu!B;-q$yRpiK~J z?)Rf-N7Sawu9%$kIp^*HggTZjz^-)eaT)<&o$t8s&(@IrlrqW^oy}P&1r4G7iT4@pg|;r`VYb% zn9bA%v$QodqyMZ0 zgl;8>rvYS)V~W+TGH@YR@~MFXfE695mF@&aLfXUZ%4*e@nV%kxU4BMg)gGP7L8u6o zYTyy zn};ogyfOvx9YQm`C7FjRWI`${%n%6;$s1!>ikl(zIMI;yc?|d!aDVmU-6JDG}mCtP@!e2zH>9nH~rcm zRQY~&CWV+Mxfe_(m=PZj+p}9!h#E$Gon74Yd{`qQ$lZHkd;4iR*m+?tEe`i^M)mI*=)=Vx~H(gm+bIgW?PDxcJz_&pK%VkGdktfc+FJ z(i1Lv0nyYgGD*SFRt9|V@04WWHmO~sQKbZ27-FAM?$mmir>WIu)R&Wb66guvyd=|N z?vna)7~j*;y?-Vh?<_M{2`JHiphtNwQ3l7YKIpdmQg7TZIU`CkM*68o}Qsipi_ zvgq4IN$@^Y!90*R@|EQ2q8@zpE2L3<^=PYq5EiwMHCo;Ii=baALoCHx+GgHdfzU8w zVlF#!vg(r(WG!PIRZyv{7~UojTDDYz=SIoh3lqq#80=J{4H>x`0<^44chyMXp|L!! z#FpOw#KD~ynWZslO1c0RwF4N%V==W>9X8X5)xVZdjFp)JZ||4s?zaU&AmU!T=yY7((DyY<3!7-I-nhVAK1$ zs${&-Qfz{TP08}|2c`;K>MxXAGTFJ997v4}n~Hg^x$h~Hz zRrnWjkHOSN5*05!fC>r137yeN3{M*5aGXE4ND0oJ%0>v6YdQOdy&xu zL~6FpvfsnSQIQi7DPg7!ol>*zD9kCeU`xAby!RQ4gz7mSrVy`eJX~CnRa6x1Wt_W= z=0dTb#qbnjzm1aN4m zGQ>5jvm#my-rcL{fg{IgN2SsJBs2tzprN_L(Syb5>uNIl(;|0+?6EZcg&#)A>p(__ z*uZd#bxwcYaO+PhXv@2X?s<8HG>-n$2d-hh`Z0G3y$jlAnMJAI9D|3`@#1;I7)32j zY&rP<VHpiKc^8eNF)l^3?-@0p1M!Plrm?OINGI8^&;QyC8BLRD2!yG(gMzSGJ2M z-`3GJy-`e1=hR10S2SW;c&eD}OATmJL@^*?aem{o+uXD=S*HO_s$D-GaH5=tR<_^t#;3&qBV@O5B# zLQI}0=nj_WSJHAxJ_h`%+Zs9vYwVS*(QxX|Z!UYW>Cc}hUPb_s3K9IM_My6zqBZ`5 z023K|{tm2H)`8{uqUtoS4My&0NWBi2$3D`nlVfoO!S zZadR~?Qey7>!NT_c9ziW_@n9D_!V&!58B97W&Jm{W>u31dQ%A2hz89BjhLG#{U6R| zbwZ`W&Re)Y-Mn0jaiqVE3@J!4GPR5*xDKNmdFW*|ii?Y-pyt?2adOXuFj+fP)w=&n zO=___VK)~wRL>stot{D}5-T4*fzF!37oM3)sM4KFhWwhY-y5)Oc2+w#|-eHiOCsaf-*Z702e<$ zSAD5H@TX3sXm(4b?g75=Y46R%dh1lIH`)9@pp9u(i zp|SfG2Z0!VbkBy<66oLvCcR)t1$5XH>%<0~ipO~KFd5;CewnfiA{-BK0J7Sd zg*Z!95y~B03Ws7tkdEhQHuLp5ok_JPi{p2vH#*wuwp)=N+`tJXz13ap0&y}Pu1lDh~Y%=5?m}tQorGh zn-7-qD+qXK)gO|F{=N0$EeipVh&T~&0*LtzIT*G4;kTH0`7;H#s=2Vf96HC;%rZE2 zE`w2!9t9Og?InWA_7-X+o<+vH^AmM2M~C_jmrl;^@LHsfag@J736+1|xMEX4ZCV*9 z)AcIgB~YqN!Q^I@$?}#nP~h{we#+0(!i@hbk}i!ufZ^(fu17IAMbh%*Ts5Xb78d7c z2Mjg6q6|=i{+}G=}_ zfS!FdC$qX6E_@+copR@NizMQv7L-8@HKJAoevZq4OAAG8?4@#@k(+qv(Xl7632qU` z;VtK)p-0C_qXlnPckIiGQs%%WJfXnsZure#5l+Q1a)K{u98s1P+43wu~a9_-7L=oi4&jHK(XNW z>d4Ym)Bkd73Z8I2$PXS8_i(voSxZLp^=g0mCx8U{?3Yd3$6Pe4S!w}LOOwk4Fk=%) z3D!(!9I#pSyks6@Ht?6PXQ8IIYzs*tR^O0};~8_utbFNeDDLXXrbKTkG9sEnW~1`QIz6ouOrg$ot6ug3?uiKK6NTq6ArZfPz5nn#nO3$<*xs3qIIf6>a< znNy39@)4~FMA(p1t%%hTI`-1A7UZ9!D9fY6J*@*-o6FU6g$8JXm;Rgp_kT_h8*topgLh=A3a8+`_*n{D6}|+TkK*@p zNjIbZ6T3QeyO#nv(8|uspRYj~NH!g!&Uk0p~_lYm_X?f46d+Wo3? zR|xAYBOV^$2{WNUWlh>m5Ju*oV~_J`fJN{*kgaWX8)xEGuCbcY-L0ZogZb+ePXBeO z=SP4?c=&ZKq02r?JgiF@cYEd8Td z_#wMBgz&ueexYiaOt4nrP)&f^uxdvVT60-3*gMoKfurddr%FHIcsoErfk&I`JRE47 z0;+xnlPXPB(%45a(5iqaPn|#vpb;rEr+K=~Ku^mDs51ghEK4o?O)S$K*+LMmg5;6) zH#FfD5}+|mtW+h@>E9E^!5G3-k^$mUQP2Y3mgB9@^4P09j1%{TCd2B&`3N}zciro= z;d28+%S}-ak>x?HBScfQL$hFb8JoSA1^o{>`}-O>A0vA`!6T|e?%dBpzP~T%KVCU* zFVMGtiagL!Fu}GLxRd;PVY8)fSvCH3IE&1e)a8{fp6+*U8WQE!9!wCx_bNsla0j!7 z9^_=PdA|DH8mFf_Id)@qg?q%H`u;&u3US;q%1uZ%ZSo!GEDk{AdW@!o0#vB~A2^bym0pY9jb72mD{MoM>|Tw%+6zcs z4Xp&Dg5rN8bYqgCcEWta{5Hm$bdog3K_mxvXrViz8s-0z>vssmw(rQBu*jhvWaf2q zb5-tUv6BOis@4rL zF^=U0?5igaZ~=nIFqcFdT6ZYpb8|&J0ofX@bR1?}`L&@k;#sqx1vd^hvLOMz%=>$m z?(|w6)5>88(RiiO#;W1acKt-?{<#ZGk2#T;4MOk}43Ot?ziM|sO45KUGk?dyjih=& zX8Oy5Hnj5aHX;luIkO^-Y)ei}Jj5u7V_bglEd=>2_1Y7Zfh8o>`C`^ys=tPe!nijc(@ezhP$ zeqkRrx}N5@$-{p!%%fn}xf_VnvKm0E4NK+KOe1{t+L|8roN?mDe`K+_^MzfgEn$r0 zcmmR^i2eOA5h)WRWWsL>_^iH2w!N0wbfei1(r6ah`|&4H9G@vCAQfA_`i(s7QPw7t zR?P5v>aR~1@%3Og8ytK1I-{*LiWBjK&Gga&`vmm9`3!Q$;Eh=1#%@W$)TkRg4`?f# z+`>`}sTUX4a+DPk9$({KI}{gza70BGOiYr3=|2GJiwXFLZ&tZOq-a~hHESbA)+wVW ziTQB;Di-r_qn{?dH9fqOFHB*wq+rvm&=2-y(Uaeq;K07Uh>>EIx?-r^>rhnz(&M1| z{Tw50cl>S3Qc)DfxBU#%{CX(M@+_Xi}uaftE>BH*s-X2Qqz$ zKvnp-Xee-aL4M!>Wf=C0G^$4bi)z7Va0}UAHpz87#8tD`nX`_~T7DRzNkR6h zsO_ssj0f^g4h`MJzo-VpIPb^@T&swskbKTqDq~=Q7x%FWA8W<7#uH&qFY1 ztI^Y+0a(iL?jzs2B`vgQ^oaA`sKHqGb-j(f9w~v>$}3#L2DT?Vj<3fjvwv$q7V?k_ zn{oW|&0vx(Z7?=bTz}X<9RFb0TpbiO`qO%htXg5x(pYus}K^W8;y z8-M8D$F$;QZREk{cP1QgThPPjSIpbQXcG&X;GJ9)-$RNR3iNb+#vn{8^&`B=jr*Mw zR?${bsqd5VigO`LZ&~~|avrA17cb0&bZq3`88vu;xfd1~D6t6CY0DVN+Ao`4Wk<}I z>t;D&R_;;Lt45z#a5Rf3EPa9Yxsb|l>HZ~y7REY>{16r76~y+hvAz4{*SbPg!jVu@ z9hg~71`tI1g@HX?TnH)@1t05}gk*q(@K`!fbfyG;S6ZCb+f+wU<9Ebs|E`tA6<6EX zNX5_Ijyz|btxNQ49go3^FlaQsV`2hACLU*E&2?{DK3uk-2cjwZ?!LyM=>c!YKHzSy zI&=EU)F;^gC>oeLw9F(adn`FF+4)Me`@^E~#@r0HJm$^kNJSGY)n-Jw z`|4=dRbtrrwb>9ZGqsOCa;N4;=f4RYTe60Jo9arV8mNwY^-Z8Nw_Hc&uvV!JlajeP7m@n)nK6B_omAA_wAVd*SGM8Z4AACp-yYED7`U$98~xLL zNbkN*9KhC{&lS}iiw>vu8mDba!=s`0pg?VqfDQrU^7C?WguyV5LIz8FjqQaKNL<<_T{hC_uC)E~yVYWl3>-6*8+$HV;y zd$#bBwgm!U1`1)mQ|Tz&N2zcpjfyLk@|i2Y#kB5NwldpT;GA1*}>XVno;yOrI{269h> ze!RhE6_1|E&PEB5l2V0ddCU(!HgV&x#w5Wg)Q0_r<*@dt{zUr>_C0sV<<4CScR?iS zSDjXF1KY+pA>&PW{Ra1jfK3IN&wavs@5K-7ujkN8Fa`kO|9!{GUzWrs*O*|}*<-AE zHt}7ml(;qY;YcmObqJjolE=1?kS!2b#^Jvl`0JOHis0GE)JgX&5VUuu#w1_<_a@-W z>~5?L{0$BGWSQxD(#MN0p=KLvd8W3*w_k^w*x57vg34;5UujW^BH-dpL!H3SXolI+ z%O{CqcK?bIX|v^BL#zh2?2-F;52+||K>UFRWGScSO9Ts^(Ej}p&ZpQkk`f#OiSc)Z ze8+~6lMs&CREA8U*6$OtR_(T*F_mlN4`E1xKR@8R!1uFi`A<-7hYN zI`V4JT02|Z05T`epWu`RrjV(Un#D{_s2|qhBUiZfMEb#GT9B=|vLEKBtMsV<;>Ehs zW;>VTq!7nc*-9-)VOR->304}~ztF;j!PqPJ=W#(iyk|*iR7Hd|GIG3T+&heCjX!{11T>f{0a_1;4e+9IP3ivT-C%{()TK20 zC6VI=&jOWe+-wofC9s(&{{2T>v~?$&6GCU|aw<$q();RNpu)oi4OFzznCLP7`-h11Zod{QKB;TJkh`JE&eJ5p&_jx ztibVW{l7bApgH@zcVu>ZDn=yh2lOy(CDd8yz-1z+zyusFMt4`NA1ykYomm=hXM`Va zYXkzfs^xGUu3HNI>Ogp5P$>Xv+B>CxYTAxW5LjJgwCXEcK5=O(3h1qdX-iHnY5uFD zmfQ)erqKco(S;B>F^K8Mxc53fXyFrzrrsHfiChi*0yAc{2Zkfn5z-g}$~CY3Qczf7 zt$=du<5yyUUNvat+!1sn3PG7KU4QfS34W+AR6a)rxu;3BLoRGKNcDvgomW!jF}Y_o4J?GKG_)M-tLM6$x_{3s z!i%e7P~QLzLsFevYZm`nL!WwLpVtcjhz~piGhJ?i4W$nDf4=-;_Dp73$Jb zuecxmt8bT(wa47(g3MAe`GD?ORU8@qqGh?s{{ZFiKsUSPf=9}F;`iL5-fbaz?kTNX zDN@<+(Z`#{ydXWWKRx74Y%_duwY+q3#6S2_0J+pC&B3yXv#JuZPR-;}VB2>2_Rl9N z?J|WII@bH26EcKfClB#zzvX^E^J<}~qUYuenBoH*t`MzC)%btvH76~}D1$_y-PXFB z9!ozFD|!1%5RIEFXA)-@RW{*9=U!U0RzB2}TLk=!e@&rW!{vzQC66701!zD(j$sA| zT~YhVg2LD}Nin)D5+WV_j(bAm);cm$dF@kt2h602V2fXM+!tf*JoC!$ygpeMjr z6N9$2{e5I-_x!6m3BWv%t52bsnv*56-2cVdTSrCJhV7#;bVxHOf^-kvAR#c)(lB&~ zbR&qAbfeOZg!IrQDcv9~CDKTXASj$Y;QPMc_d9Ey^;>5x{}R}HKXKpBb;or*h*shk z%2s{|)`w1%b>(&vUlo|rl^-<=kr?8TZcOYcZ&oEzH5p7MwsVb zAKf%?uG}@#VgjG{ zx4vQAT0eySVrkbYQu=eX$DaDWcw|2C?MblW8lSW^8Fq=+rUi>gal)9<@ zEgSUUpG|t#qT3*$Biku%J>_AqY_;jKhZM>Em~NqzyWHrjDqV?{xF1?dK5Cl=_Lnu^ z$py|{2ns~H=r3#NooSJ3@)E*4KwWDiT9h}xuk!fK0~mQ0v*nN5^xb}Nc&LBN+q){I z>z%bTi*fn(!7fq~_22AqBvO5~RulGpTwG-BWCs$LW%O9T4o`o^wMbDe@x1mPyL(c zBA|%68pbRf?MFwglnCC~(`VWb-2Ay(>tyGr*>~pseLbfsH6D_!FoEni(wr@@Yh+O! zD|J#w=ZetBY?|2>9un@(bd%FGmgo_N)7&B*fL2*?zsraxSsRp&B%OPM>Nk+b%=^iX zBjZH?_s_rzM;ue`)z#E_iMl^ueMPda&A$WE)4o2p&}4byh440TYF73S&Zgg$hWoeQ z#GmX7?RsyuZO=c<3yaOTs|>c8S}0C(h-2ZHrwThi{j7vi|0^#-M;~w|PgT)3XH=Y6 zBiYA}ZOqPaKa^;on#Cf{>e~w7rn0K$`7%l<@~1#jXkCVhIbd4i=2b0cPhtHlUv7kx ze}+{?@++X9+5<+`(zh^hA-n~AQv3Gc=l6e+)y6#MlpO}b?@HRX*>vs9xO!8i2#>*$ z+>8J=H}7ngpedW}i%+?fA0zGy)%y9o)dX=CNfX-%*Ckz+VE(vmUAPSj{>T=)@_j_H z$F@iRs4r~VBlu=hm)J^IsI(=_&>9dS_IUh~)8RY=V@ADo)}q974x^+5zJD#?&h7Df z+on17hvF#?Xru!Q5DyE<++BD$fyALQW634NYd$0VfP351dK#r-m^h*Cmj}bMN4CfMxFK6$yK^tFt@n`_(ALUyddWNA z4A!FiY0|MZ7QBnRjC}Co$J2#Jzb!j`cvGi1UXS_x2niyk-;)Aj2n!lA!`ij;;tfGh8YF4>o3-hFismU`n@PGuPpbe9GvrDG1pUP(ZkRzP+G78SpSG^uY?6zx3^B6#D?jk+Tx~e- zaQp-!DnpV)OJM}2&&Fq8xjdQwd>R>mfPt7f|BETEU2pVd6@bP1*?aN{j5BgywKgIp zzu!e{!2tbSbX;+JytD(;M0B-|=u+*taCRi^#dGv4cAc)gvx*vdiR7hm$eZ^EjivH2 z+l%z&``3y)HqFJxp6V$>=+Cmx_x)JpdwT#^H6RJOWWLSP{GvT7=gbxptHgW(Uzk$dl#yx#ZdwZU$pcoW#H0?%V=O$hA zw)h$h+ZXdb*2i_}+}*kRf&66qUbR)J;zeBjZ{l}f#paxiy205j3!c@=q6VYQ^?J zKOErq^cbmIT;oK8JW-956vlgE-S*3{q_ioPk_~AfB1SKeCikH(oY-BU#wJsr&&g3l zlE280fh}N6vbN!US7G2V`cNzwi3Bv?^gAMG;T?-W`%UxLeDz4l0z`*Bf{tafg`i{Q z_e0R1qL>FT$YV9nxm2_{ccs0{&7JeSswP7-O{glPeEhz6){@gDvOTyBfF?;?(UClq z2!wT&RLVTgpE4+kAZYFbA@~3HOHg}3-vXe(OHy0}#?s(%A%L>gj@eKM{Qgman@!7> zFPX=lCGK~9iPoJeRWb&}IiR|z+Ycpk_&XpGRLvS%RcZu5vfA3m_^O~~nR-p{{ya#U zWfWk*$OP2@yV*pm;FhWVmkrqi{!JwG69gxQ^TKmbf3huGoI;78)d~22Z7t!>rf$9X zn)mgq42*Nn$$j*e)9@Nz)u&&-5 zKd`>FKTEw1uDxImgzd`KO$72@csw3Er-_b zk3IFXJ!k7J!H(Rg+^Kc%WycVRmbKUKA3TeF!L`q1JDrF5;&$gy8@UYs{M7c?acCZ; zWQ4HQ&^9_%%YWwM1CdW%4-AycMpcUe5+@-7Ng+B;KOj0zc?iu^kY6Hw3oIfgbV+#a zd;am2&dhX*@=59lA(+-WU1!pBOlk7wk51bVwzx5-DNV1)=8Du8WB-ZLr6kX*>%y%O zDm}P)qCW!VZM@gY!2Gv&IP!D9EyHaD7P0q#Lv^+_^9WS{NRE*RY7c_%D>G6AG>=CR zT;ao+;ABS@w6q+q@rTB6CqQvSbBp&2&>JvN|L2Sd_@azITDAt7z*W%ot)x7Zz`(?e z#jZ_;>#P955+b-d?b~RnM>QPaQ>doTvR~#rJu49#v3*u z=t(yq`SbMAKT*%u25v2V7a;DTrC*+<;PIqod3n`=T+ALOsL5GbDc@?&wP)vFTf|E_ zqytdrf0X020sD!!F%6cPs}8WEX%UX-Huky|t)cC<5DaU5eUmLG(+$OQ9SI@+jzO2O z9CI!M3wQ7P)?T#Fd80VO^elJNoGD12R5&PWWd7V}0Q_zEW7}5+T07J6=t3@!Su;Wv z!c0Ws?_Vw{LP01m?@GG|1K_hFD?k2UB*Msd+{qbpjoU2O;lFU--wEk z|1P2`!y)ZrhW8J(OcuL-d)T)sFlhoNFC`=%-bT8D;KdbTUpk z_VN52)Wy2NKm}UANe;SE+=*Jy2GWPygmY9E7FjQ^X=DX0Ob+akQ_2$#0+f8rTY32N zeL9}{H`pzYykV9qbeC`sg)VQZb;v_w|^GQbMY*fQO?iiTlDU3g%g^-}_e>tYo)yRKAM-l7#OxL2DK z@iEm;O~OpHwTz!MiuaA8S@}qPD&kP#&a#FFQ{&`&C(URo}-E7j+*_vkcNfc;p00h|K!RXCq1lL9ai zx;&=)c{+*6K!#CfPOg9&Qf9fC^FX<+P*yfPwggx(Kr;p?k|A_|ys@4E{TLjW5V3Ec z_VsZqJijokpITdC1^&njGt@j+WQ3SbOt|wpoEyKH-PA7?6?uAnfi=${)6b=AR@fJc ziOF%&9^WLw7%h7UV2s(9Mf*9USthZ={YC*NBG5HDP1?4S2$*K4ahru(;sLxC?`2&X;}A(N6wx%sg`@-gY!LWPi!kfo2827 zgdvUxuISom?`5d}=8oT$7d0zJ0yaa~g&?D^me}5II1$1l1c6WSi!32CIp{pAaqsYE zWjpj!q}BLDJdSAsId|5uor7K>>2qwQ|k=xX&+N-tp`Oxu<}dvNb_ zx}Um5y6v-DlToKxbwlLki;q-Wuw+aQXhio<6DB-Pz3z+WjBOW$qM?LMa^Gy`<&dD6 z9-q8)`2A9Ts7>Qg?+B(Aq)HbVK=qH@$i;J}X(DOFD#6tN{_f8l`Iq*t@`f=Fh;vi# zX>!7rJ*SqFCA{sbz86#8&K{y!2O$!~5DFb(x)+G?Y{6l~k8bbbWu$0B`SU*WhDpzr zF}g5j?5?$mf)P6jv90+h38IzgB*^V|+*lw0`C?SU{uGg%F*P4f<=YH7By`M{z$uc@SMU-!9#I|81w99qxl)>HPivsrVMkKhHD1FU+`4 zyAA1uwpQnzg%-yswmme30hs}W9Rqyy+ZoE>Qa2_F=%{@dUu7{z-oNh=}uTRPlEMW6hp=42C)%kUTGDblKlqzVn z{gF`%t=QH{Zu->*w)&TYr0H4Dn*w*fr76y(sHBFIZo9p<$7iw79^KEAoK9zEP!p7Y zR?a5uRwyRC@%a|8Zja1=1@%;Xn%vSH@oO?0R(Nfw6(fAq*bD9Ke{W0<{2{{C4g#HG zh+6X?vglb=5JBS^CZHAUspq{~A?;(xF_8)H56yNgIy*R(Yj>8=YVi%-J;pF67YM2M z`16#ht|yi&N4&J?TdD+rUw-|W^geVeoyq?~hAVa=F!@J4AJER}kp*lIp+xbw+wS>_ zmUmQKgq|j927vIounN0E9L3CLpOh|qd8XL8jq9oVHSMx-@_%xknHQH$5`^*Q|}=`_Qy1E>rXL6$1n3az-htXQF{(F$J|A9F@{aiyRG;|Rx9sw3P-aw z@&>GIu1@22UYt|}on7ywYLyu!FMA9cMrCvU|9W*h=5Qw8X0zB++{b>&<${cq0wt!$ zmo>G^9(bM`dntj^TKxoaZf8X@6Kd7a*#M8!y<2o5hKWmIh^{~ z)7pIhG<(GXGaa{Vp`-MkCzkG!Nm5xnIoL&f**JM~Y^*b7azWc!%w1O? zpal}*Ip+ElIA~~%4W8XL1)59&k^il;{3A)N#4%{yi)aZ1^brUifM%5*@!CaoPFugS zU0i*H6&3rdVf^Q;v5eJ9(ONsZ!3c1;aimQ)zGzlyEm0(7=J^jEK^-0V0bK|n|I$@F z_!n42Mh77B+e`>FRs!@|+ykCZel`3|fBHbcuO)ErUOz4luMO`4RVz37`>Z+k;pJU&jgO$o=XHIW3G(X9fI+I;8GQF^pQkn7!}=_5dp@n8O2`SnGpa!8?8>}R`l)g(}k)76ViLU3VSBK z7E&7A*?UvI6!CG0Jod4e3Zhlz`>)s)IWua@j379KBWVKg_Rc){ZlAl15a9&OGjNTX z%lahcsd;McmwJ?WhxF#P`}a&YfEJ6#SC=AXUq|1Pld0gj6i2dxHvY#K|E)m+S-w3# z-rX^j_MdB8(vlnQuTd|%`a^+Rz;AtzDk>3>#6<{IbfrT-#nFM>m|~jngp86xzqd+@ zq1dT2c)1zA4DB;^e3}%5G^^6;c8tZ~r9i2Q`kU?+CLDe?2c)C~KiEPZfq*t(kWy$< zLRUX{RiQIkj;82SfLP!(O)ZI!Attdm^c>0J^%)~|`o`Jqs#6DIULf?67~m{?>#mX} z2l-XJomsp&3kY$XGtuyOU0(B8>&3G(6Eg{|XQSCZTX}AZU&Da1d?aF=MA~g*1e+P~ zCb6C7hK*Jgd}z*h0yz=WV>`(ixdYTP%n2XYrOpna9N}&cap)G3G3Y6>-78jY!&}RI zM|g)j6A6QNZ6^sDH{+|`%Fi3?4hiVv6z47*CI$I@Q8`wJ>_&(b2zp z-?f1G8Otg8mSt#e$3KQq==AE>_Z@F&$3u`j6NCdJ7Tf5u0sR@8h3VFO0Q`3)$_4BtHiH3J0~)sOVzCaJ+F zKl0WqFjE3!GZptp#3KaYrVusPR=8nUdTf2RNdh-W-%0(`HoNV-r;U4e&mP+s-4ks# zr;k)hxctLa(7pTq`&&r2x^C`LXDO*jK*yr!i=!UHj) z$#igZ+@L%{d4zF|BlnewhTCEXdT_1?#6U`qvA<(rBEugZNp#%nAnqvbph1>D8d)uh z2fK#<4hy_Os$y&BL>EV2B%VvuC8hDW(zPz_33Vd*!_}qz9yHH!x6SX z?{DjfK~AMXjU-^x83J45U3>_?b+Rt9cX;}svDiOS5*%2HPw6E&Xf@h>^5EAiXG2mz zoYb@%j*MGcxhtr^miGB$eO`>tvYoYuVzWl}*Xw9`L*Elh9IEdP$XYp=?=Wrw(Jx=- zi~pg%0^L%ISfqi<9^S)@^zHyjVuDglC(h0naifkq?Xgmy)9~I8&S*pexXVIEjyX@Kk%^o@rwNt>7KTsK#;F7efgV5_?1sT9)ekgCHBA1Cv1F`50jmK19ui+Gx3cxg-{1cx_l8v1d4A6mn$s5Zt z<`2)zCl;XMeSh7)c=g zUHmc@R{Wn~D!|-1ni@eDYN||9Ngn;m?7=-lon=__G2bLQ)d>OK5--HUQ@R4S#r>ol zyO`bO5f?p-k+zOqoSup$$j*Oezqt()hej};F%)tfKM$s5?>TGxp&=4Na-XfSdfFJ8!tqr zJswA$?VG7&+f&R=YW<`^b>b@aFvwNBSt=Wfl(dy`KM{VaHw`c2^<)pte27dqNz>0! zNu7fqWUm+QFN?fkRlmh2GC;HgRA#hjg{I1GmoZWj_pf`zng7o{o&`fs6X_iO6f3r9 zs}Igron{{HTHM!l>hxBnZ?VN>eD#WZjvnSYVfYlF;n2_w@dy}AJs1_9uhztNRRC^| z3?rae%eGHt4f`$5ck$2M9hDs71A;>9! zZdi8m|Fqe!_Z05z3h)-7zup3w#P%=VqAvUgkBRxF#Ih*uCegyV5pjLx--j|=-$D@N zztH9`heh;1&;`f`h870Y8_)zRLJM#07ewF;XcVrk0!Hm+lmh5n zBwcJIPrIE%EXfX_!zuhVqQgl`=?)1}BmeVd3NZSdFfHeBAg@*AQXpCMo#4bBPAtbL zUQx_+dxCd^Z*krEUtD*WWkdZB3?>9T((a`B35BiW0BuBA?}8IXtQ*wFa7>Q@?In>0 zJ!Ku+9P;GbFL^|GglJqpxt~{ZyWT)d=9VwffBsJ|z<+pc@Z&lC;wuJ7msh|@zzv!a zk{`A&dJ6gPD|r)iE(#%>59oy`64=8T$Oh#0qCVtYA}MMvjBtq~vQ^7ff>p^?w~oDz zvB38MPRP~MyN3ZoJ`*)ma9UrOPsr_uSVkbwJU7o@G=CefNrL{#+P{O#N#d8T5+ni_ z<23Sj&~+4cj3Rpxv=T+OTm#n?ka@)6p?z%EU?4FTv732#IQaJ$6xNdE=0s6XJ?cxqTtUF7HQ zJ0VerAY&9qP%K&nk{^0Bs2<}Hww1;()^NXyt~jksAK=C#4FjeFrr_JpVBgXCP?Q7u zJ;W@up91XCG%{ha5a>Uz&ANT9^#8?cz3x*h3I?aZP$|eEDD$+`JQlqnQ$_J2WKy;U zL@B>KB<;Ur`HH3(Cg^toJX;aa+L|UHfHl=X&}D{6&C4-^LcT!NcP^5OMU zrHIJqBO$_ULbq?tMiCdZm0)HEM*sHO-x>SLJYF25rin}cbkG%73&QjX<-Xq#%lt-x zU))e}LDE~Uk3$2C)yrYmu+)*Fz$-y@)9swd33rR+F3;Zcy6>`CM}6PF!xVdU+tF;f zAbr(DbFt5BMOi-nNAR*b>ilr(`VYg)_SlU{Nvogq-#N#bwNh^qLP&6#meV#GQpMxz zph$82emF;V@#rWjz$CtXU^uK2qum(yH;NE-{Eh-2z#Om+Bo6%5c@p0mNcXxcGw+Ij zHEDGVw#o#APBY;?iNH43L`CE~`s4t18<6jacw&P0GJceUERoFI`<4+lbS*BbKYVr7 z*(jA1G&poqt~BHqfAsQqP$1utc&G=iWn(hgDL!Tt7}v?mr4f8}^F@@{Mz};I=7tr^ zB2aS9YZ!+*NGLB)NQ5ksRRe(TDPuIJsh!ytU22tqhO=_ZH&H)#(%nnTL+5;N_ONBkS!%N3;AC%O`D*Eg=rCp-)MDoM zxvB9em{lg@(~^w+h>*ahZW2$HU0|PX|8hn&$II%Wl8j2syuRJe5HDZn1})JduFNqj z!EJKJ_;ksMGX|$W24@i1x?i?I6%G~*ZgZ`ipcq!|u{|SM@yI$DO&6!8j8l{3-Klja zk+gp@X6{2!1YQZJ4qi{0s&TIyvM~3b{zdm&02*f`?5Pv?bP&K#*Towxoy3?odxx44YDT=_*yB}0t5!CDhDxW z@36Hb9yfcyOt0q+67<7^W3<0Kukezv1TmU`mSm|93pvL6hwRx@{403%Fo&%mD#tIr zk3Q((z!DLS&HzIOeI722IupQ6xGAQVim*}wuu>#-Q3n^N66dvbPu)aYC07kR z@42vC_k)Z^7#~0B*3*0;{JE*6!UVZy?Y=Hy12-Dl?@<44?X-tH{iXN&7&PYs!_TV% zhUVz&*B6t*z0kd;Ws9l3GJ4kScm%- zz-Fo6FgW;%N9!3L^6p7iR5=|{70udc8b$B1E8e%j(4d~ks2nZDh06jmasMo7LB#GL zDR=73d$?>Fm&=X{fz_gwKIBIx=0YakCJ5va^sua-&fRHACw#K{U1)s$+9hX*H0*}1 zI41>*&5U8H1|G1|orEHfG)kZJx{NwhD2SUvy8CC9N0+SGc_`_|OQltBzo!g%p{X1g zRphhqXpC>4u1V@Qsj3QsWFmH@TNc%r)FPm7^S4D^SwDZ9`p)@|3^mH`!8!pgV-IIV z?jf zz>Ffl76`M64pK#=fE$GS!TeYHndPv-j2Dii0+EK=M4e{H`ZX+GQ|nxz_^v{6Wx+;c z3^IpLmOC<)eNlghW1%KHXSSRqsn62kfW50)j0B0`CKEA-k$o1o`PI-iP%7kJZEc7-MBJyrR>6r7UCBI(}HGYqOYKMgXJMC3;R+cFnrnURRr^o7g?i!Mi3IWa{L@}?+VpG+BynE;tS)*#fveiyax({9 z=$k3tvsBcbIHv|KRW`GWBRg*p(TDTJzOTCH`u4%tmUq@KmyTi5pfRY2kdmngN@@_4 z=u_UTL{!#_GtplwcLYO1Ps_bl$`KR5`b5I%cKlo!cUC^*%;U1MB(rK?f%+5mnTbf` z20Sn0(*{la`U4m00R@WaQ>5M#xo|HR8d+D&nz^X(q1UT9Or-6mWQA^`)^IR!V+{5e z^gxqvY$t1qbMF47;?-o=q6;QkEAEi@-jOY9N>+r5x?KxOt7fbpIYag>3QvW#!`W*z zUX>0;&Q2B>;YN{q;L)13B%bPj^~^0}Vl#Nc>Y^0O9_ZCq|M2WmH2pMC`dj*L)%Si^ zKeCoOIXEm?o?lpL>$1d!QCCVF^Y>%R0#lqQ$IH5%BYcN)U2IT10tY3Nw#x5|I1A%6s_(amI<7dh}XJn&ZY4Y?K?--I!8CF57oS1_F? z_+8##KTc#({rvt$z>X&{{1c4i<}|ge4^_ZI7$;@)Gh1m%=4r~HD`Ul^9~(V~**$O- z;ofOS)aK-e_FZnBxLk0UcJGe0a?WEAbT&Efj2-sw+^DI5gU+NgZ)lkSzYkRjIF57v zIvgz5}$~B4guBn)T;;+du zj+dF_3UfxChJ8}SLuzVM5?*j2ML1sIpByo>+bfnD_I5?YP!awp+9WsXYk}5I@Ls}j zoue-N2*ZV*_yZGcdxb%UuR2GSyevjq{0M>$JV!U?!sG9AKG3ReUaDK^`@Kg@Fk+4$ zbea(zVR8v$v>EIrp+!;98y47MS?I!)0yQ}fw2G<-=D2-xs~J=FhTctw8{f?<7cu}G za1nzyE=ScuINzC&+2Eqk$AqKjpxwL5qS& z`L(8Fm61Tw?yuXg$_88yvqIbQbsc{FGdDI?#3|4bJ1sCxyW`y)<02l^)^UaOXa2)S z(hF|5<4gbqW5yw*#&cH`S(WgHgo>%D$);SZ!c61Pi|NjDy8lB*_8xG42#=zeuOedbG*rMI^mh4F6x z^QSGNj7{GSl{@g`-Vk~xr9~@+ReOoP+xVs z7n--&F@|EUM^&MJXG~myT9~+rjcqTaxA2p1S4~nBbKTHb97hRN-0+0@yX*?sU|%R%Zzi9>m-7{DzMzIXZgZtVb{Kp(He)rhW| z6rY_m?>#UR&VWBMk}7EqVnd7SsVV$px)$A3SyJN1z*X9rY(;58!$pHq8`S&p!Ha&U zaKH;QA_KQMP}+)-n)O(SuVXWEVPn}NFQWcP8GNCv9GN=~f6 zJ0_&|gZ{Eb`AWhI&Q%FprQK(%LmetesUfp$L{okkzlohuosA*_7(-BE+hp+@<_>yI z#i>ectv%&vKIWKQVKqnx;NXaVu|HEjdkK1sZ8PD)9q37HIt4IJ>d(S@Ak8!0J=aZbf&?OXY_{Lvj9>S-jbLBd$W zNJhlL{TXq5WuA0L52E%gbKD=^Sxca#k&7Xj^5EqJvw#V&DiiLHbM^|CQM*ER33Cu7 z6LqK5bvq~J_TkICUJP&jRsT;h&s#fNIr2YGn3fn#n4V5|e5K6J75$C|vPS9))_=KF zxVw36`NVhpw|hFLVvhB$l6Ii+Il9F?7mTAVqFm&aO{mSK2{WA)ge=J6Q!n@!W|d5%V6nzG}#hJ&cP%s!REi=OU8| z=P~;4$FM*Bu1`1!U2DdB*M3~o@?%- zSF2AX!K9I!9~Fi8Sn%tE=*0+pcN3Eo%rbkK>@ali@6_o*&H5Og_$ElS_2%RWaYsLo=vVtm%4)QLm7E$=oj z0gK!wc%e3ED-(E+pGvpb=5_J;H~kKF(+pWqQK*YQKMZCt6BMrT!)@*IeO!mw*HhNq zvS>4xR@_L(_x7L|$WblhE4tr!Tl<1bMszN^XX}$}PyXbXD5n_3S(?uEQ`StpsNL+J z*4RAUIk_SjHuMLxG`}17d4Hx%`>N&}>txZ4p)Oc_52gWy#|$=mcY9|lBKeT_*S+1tDOmyVb<+K$` zt4MAH@2+~lupHwT1Rt%Kd~6PmXR}^bw=*2RvadOFZ`6!#Sm()|_xx>eNZe0XF%$$# zR>-zRiJj#W+N#SH6?_ibAc{B1Ria)Y-#%=qjy1_ei2?`h;?nLUY!eisXqJCcQs-4I_I14ZWX7e%24wrCVq4cINf2Tr5}Wh2HDmQ!}L<)uOm%-%H35<%!0my!C{ z4JkDFs3HXACDpN5BX#LIp>E@g#K)V&9XdgE;g{`H7gCK&Ga1In_LLC)9~otQBqoEK zt*sX^-(!X4RSobjU<02LJl^=B{CMTg9q33MUmrf+OgkZI7cO2zG^<4~HRQk*XZeH> zC(7cL94d}3m{l~+@4T*%Ph)Z~6LMAS$@;qM;%6wuRtTJ@R z+!ywF>>m5IielN5{d^iTDo5V*Q;VpB{#laYPkZN4wieC?Yfu3(umGRFu%rBWwKYu$ zVGK>TOJ&{}(yZL>)!WTMjn{*Q(WMSIe3~kD=W>LXV)9j!$T``&n>@qYkN8z$q~bLC zMp5YjBb0%@w%ToUgyAVNXgqAG=@;JXO~{(c136t5`?t1j?aw?>(+n0 zLo#;Sefcb8s)G>l*Xw~jpGS54(98WLhnlcjd8|k=u@P)0 zJkYFamkME-&JG^*kp8o$zL&T7#IoSSkU*1NzpMIMGbBXIeJ8ihuqzJKYq#_{Wkc^& zd38rs#hmcrSvd}N)l>$bwXTJ28(3`dHf-y2mVefD0NExS$^TuG}f?k#10Flk1}o zBiWl)`;Ej-FR`ml{YXcLNDc>F4T>)a3~FS27;d5Q4))+ftV~!~m%S|(Sb@DDU6V8d zA0zBO!A>+Ck=SkliZW`HtsRELZ}KLw=k`QTfZxLM{q@tun+NvpZ zmRIj*`nzZ;-D}dI3kx#oJPu}BME)kaFgbX9hytCPg(X@7zhX6(qlMLCA{7t5tqkG# zxfDi3L#^qQ>)vE%H+S*^(yX!oG)K+Walh!tD>=wLAc>b8#vJ(@lOtvXN9uhi}y zvo5b~d1i#7oY||QSNXlv6Zy?@erdMk5(e2u;7BPwaw54+U$R$y0a?e-Fa6Zz*Y}94J#`43(h08V) zQ*Tk}83#jjkK1-hUxm?s+!!cio={_1XK34C&!*Zw6rBiy8W~i~I-u5$GgE*4f_{00 zXMgl$CQyy_q+K8-08dxX-d#1}@Krgj1`|uPH%Ygn`NlCWRiF%9PC)L=jFvv^r!8Bk zUV6okk#_=0&4H7fbWQsS)6lgbngrzviby7!3FWgAv?!U!b36>e{V`Gv&`ZbdO@%7v zA4S#Hn(6y~LNFaUnFJVhRBV2K_VG~hy&Xq>Gg!Zf^b9`hGzZP|p$PZg30s_~+WvU3 znfp-wed_9rc-6cwthz?eITkb{3x>DO&Mk=TaiYFnwMB)OsLH!l%vgfOR?uO8X{XFlgc9 z0I3WpYFNW#HMzKKQM?42yp9f;hduwyh%70nme!C}UJq9_?1@t&hd7MBIW(*@$yH9N z&RZl^@+20-!=h6EZKn&>)Q_c4H__FTYkcQjt`vQieKE*qT-YqHuV4zRWqv2qk>-E)v?IbRTN=z1-Gm~=O zwjoV4GN?1Y+7>WC-m1GE(lkycP1ERNtdexWjnWpMe*aAu%c*+sxrJ`USlZ8*-a!tw zoXmFGO}q^A>nHKQ4SR(-RUTHDgFZc#3<~KE{{@^iZ+VqM@`2F%J=K*DRH50i;Ovvp z801+Ped@xFb1_;^571>#QU?c)nkmn6xvfjDp7e233GO|mmS_2f5Rwb^0|p|URr zno*CMk0F_NoZFHyzZX!;WAkg6<{Kv&pf~G_7dreD?+swbg2~fE8P5 z1!)b#pO5Ks?!UU!#iCODv|NJOVeOi68dhm|It5`#ha*W8><39|?&(LSE*H z=IalKX^{e+3+vAx5j}o$&!$UR>UH;%Uz?@<3R1M)bXqe^Y5qi+B7|m>>EoHN?UeL1 z)u^(cvHV$B9}tmY7z!yWm2iJ#phFZtJ<8irMR`B7=L}EMldK7Is)nfS%=x4x8{Egh ztoXfRgDCGpT8~)M0?L*k#ZVanrG?RRu((FV_vf5RB?aqg{K!-$D}PHwQOBWG*;1V` znO2flHIP=R2an&7AzEz`3561sX}q$FE9KY0=(7-0fIud}%+oWe*y`ms-HPB4>}r-F zMXcYK{<_iqC6i!qlX%NlV!1%uh;ay0=;-M(i7;e;vZ9}eF_;)v{49y>Gi8|th9 z+c+E8#sx%js8K_JzyHZ050B?eax5=M+3Rc5R%=(^w`66ek|ZPE)&?oG7lz0teA+ls zt^5!xK6_%P9BHO47aYG4C}0?qrP9F0t@6kAb-n=E6rr#t(~d^u%lCPJ(2MD(0;J~h zd|wSXlf!-fZ2A+QW~UMTN;W*%na1X`SLcn*aRI+QNcW%^x!U6GP%A5!R!sI;2eLrg zppamDtOeJYaW94lG8&B&y`YdFoju9Q`(PjKCt1YE#uXj9U@&$G%Ue&-VN$&eutVUwupn8#aYo(KLO$ecnq|jxeQ)m$BKU>y-u4 zX?|0Cq)VV7YgNDGGxgjg9(lpLY#~=3R)3rUSP?+x{tX&XyL`v&jouG@JsgZ&6sVDy z;6anzhjV%bM&l#9&5`Z&rLB4g?;@z*z##qn(69;s(F!!_X=CI~oht+dtyxd8RWHUn z=YG;{{ds1e1)(Wn8j?h-d)B)m2UmTi>pbN;p#*xPFf6~I{^u#nP}n!#Z{lHb0jQ$7 zrwUy5n(xvy$wXt;t_!l0zr5)dEokIaw~+tsKCtzX_D@R%R;>~>A;j|`5ZwSOrczbN zKYAuN)t{vAlBO;UQ0-AZ`Do)U<@h&=TZCytnIQTOPY^n6G8B}fznl)KNDleD&-a9y z9cF>XH)EO5T}Lk(T8-b&`UE6dzB;ae!{OCQpY(QjM;nGw?A{(BYPt5503hC#4=M;t??5g#>p14OKZm(L5VH%A@6u8qa@#CW>-V74=b)aMopVAaxcMx zWnzW&A)WkuCV-HJOKP{_-f#*vD*1#L#ftU6zEZhUD=$7TF}LPF7Pv}yM4bb(PF@$f z*`Cy{P^nf(p0?9YD2U_mIujeQD^IU2J{*;2$Ud#^-J4a;(F8@9aGL_jiR1j#(^{cK z%Pw{*bF?&XuikLd7{jPyUx>qVwcRg!y@*_si41YPCG?HD>ak{MshoihE>gr^g#RDs zn4Ujrl8-XZ%#7b4L7+lGecub}Tan(!jH4zLdNUsIU;Pz<0 zd8yzrai7#e^X1fIjQVa%^;8nux@7+Bb&n>G5~r#%ex@e<$jr?p1qtuIZ$3+kl6gVh zXRm^6W^D2$?j2~v5IxAx)_Kg|B&C1~QU8&tUX~*NH6uCWDzf%#Gb7h`)HVErpc=tU zK1-Y5Zf~tdKX0mDM_ZB>oT^1fmXQ=L;3_0&Dp2*+GDo$2LDVY(6BH#I%8aqDcX=eJ z4)W5qe~sAj>TQY$FP^&I+-0oF3FmkjDu$Md)c_n^?u5;nUL?o&7FLMsQvH)nosT1e zq6P2Axz?js`8+PW1E=g|q%_;jhEGfiCV$TDXppsTu;CkDCWj0aCJXWC!uk(k$d>sh zf~-7oB24+u5}k52vXU;IHnJc}3HNDG-J?p}Usqx>e#?y3O$d=|`+vH+4xpx%E=)ow z0qIQ;sZykOPy|8~RJ!yU=}7NABp`@@R4Gak6ch!KCLK(a4pI~asS(l8OX&T-Jow){ z&D^=OGns7e?w&p8?Du`U_ap5v!})>f5Jurs_`F#+KekS!hrO*J*);_`^!xHtXceHRd{hPXD;?MBxK&Ow@(B(Ez0VxssCdL+SJheE-q9vxT8YQJBx2Rm5-RvQyqPA`eYm)H{eKUl)M0kkwYGxA!Az9}x z4a}GwfAJfWXjg|-KC{UgKQ93mCI99}0w#|tR>(m~XJfd%534vI*0%Z+{q{TMyk@n* zl?0iv0_fD9b#Qx4nUmS zbp4cvIL6~Oq~f^z%%Y}kB-l})_?B_XO8RH5X3i4Zhr^o*t>g}@X2b_qXF893HARd3 z3>+L}U2KC8t*2N&B}n~p=G3Ag_7~D4bCOlbWdbs9QY~++jLw zE&m|$=s3Bs$`|tdkoS+;f{fvx?t0V`C>5u**IFeU%eJB#9K0i@`UTad9s~WSz8NqW z(CE%{(t#4L^Fw5V3Up4r`>wIp*?qwi&eUWtE=O29>~sp?!1K3*tFndh>=|t_k$&?r z9^~+F_``vKs-?Xv|1e}IA1FU@n=fbofht-7%}y{OHrH=0=&AK!!N*fyEwEm#yV6S$ z?%g~*>D>>|c=;xz{0eU*d^SLpv6`)$Uy*8-nX7u$IZV^y7xgzVjPu90%FPe>-ieF3 zj#lFC6W8YY3`O}`IDbicf-a{qi_Y}y3*Q5zNZ9^Gpy+z-DOa4WNTvS zGuEy5q^Pfz^Nhh(SJ3HvV`l`BJZS<^Mqr2DRmC_p%i*VFel-r%!+d6zD?6b-DE~2f z8Wd;lLR8%moO(FD;I)yrm3ZPL<>Kc%2MwBeBsmTC--fIN5_a*e%)aX{^p{rkCUT+n zbf)U%rk|$**X;T!6dK#SslrH2MaL5yYQf!*XQI!i&FmS>)HJJiz^fCb-H3^Q1{Dnn ziAA<;vV}-00JI(|k7Iz+`|u;uAay?M68T3R2D6-n>0rV7@^iiOjStBB0WLx`33xL| z`z+#w+%K-tNui24PZQknj3DRc+Vkq-U@q)3o8^rya9Q8jgM~Ai_LQsz?l30WAQCSlIVvmp zb3bpRadYU0@%t}xby`t?aT%2=G0~l`fLIHeZs^Lt1g|^+>`=n1)8S_-xazhW*)C3C z)ie(;f~`z;X?>>7W(y>Lr%w9pKHv_yWPHFH-n>5P@v>3N__t$I7jytH7cAxW2K|8~rYwAmFG$Idu6fX_h z>g@Q8DO}={GC6TS|G8$#z&?b#(0rwIqe4ynEgFZ6yV7#F71advmHs^G%{O+>o%`-- z{MLS;b6r{8N(3neVANev%0%bp7c^o0U6Un-QutRQ)!u}#JTY9ewr1?O$%b3ib_v=AT6=5_Ip)|^BiUhqw; zknf|~s zX6y)+%HLu3sR|TF<#8u#l#ej)FpE)`4sn8UU{R`P(YCgA|5k8T0C12Iwq(-+2vKDs zzj+OWzIFvez5Z-=eQqxeRj*uH_bxr=pcLvSh`JUT-Hy1Ph1QUlquJcFQTDGsHX^|^ z3jW!66cddcy&U;l?!{Q#JkzouU?tfNs0P!i_ulfSj`h3*vc)bJWUrh0-$!$yjeJwL~k`xewx! z>ac5RF)qd(;Q9T?Mjd2aLZ0(r zCW)JnY?u`250IjDz6l()RrW-|_qp`cX7`OTN{WY95D%X(AA8UdI1zLo6?E3f%dUmt zJ8ylU`s9bb3bU)PjNUpZ{?m$JE3_lzJ7TLc*hR0kU0AKocFBI(_Lr#RU_4tIpPEg! zly9Ex-MTDvF$W+JsDOH{^KED057VZqma~xsH;eGUq!XV6x-0uw5Rj>GZ&N>DB%A;4 zrVp8TYUUtC;2R6az4Ck~`>zGh#oNfx1tl#BFg85YPX&|ZbXYeS~L z9wg-!oDeIGxw}t+iL?j}H$&PGjT_^KW-`jOxM-CTqj$${^-bQb?biFHz#y=HUEdG) zcG)cKW8O>QYix1?u|jVGe&Dp>tu*|qJ*za4vyRGt=8~!lt=E<13P@rk|r(OBc*x6<~F%~uXx#a84l}@>tQ$1t{^j8A>t$O zIC(D-u+zYnRw56aM**BV#1^0ZSA=rbg5jc|XnR;_FqF>cIKy>U3Vs0|3g`0TTMfk; zV~6hm@|Tt=&?I4)lD$apSB+l@?(L%1c5@*@7ixnI?<|lt(F(jp%|6tt=jLkg5i>{Z z&M;HWhT-FrX}pcg+O4`EA;~ZRbQ|EONCpq{BR@sC7{~XGI7U_c%y{3T^2=y2%ek@G zV34Zkg<1;2i`Q6y;gIyJ(EXg_ktL<3Lo+BjMGq{(hNnbF-umg`zSHhNU?qhV0OX7g zhL_3ked?to_gTrTfmiw4^9=8%I)(8yhh9va^1X>8A(>&GGQLpSZuN93%(Q(QPseUM z`4b<~eUNBxuxGeQqC^(-A+4V!5`suCBp6-t5S@H?y!RouBXh#3jM~nO7+~i;lLev zS<&g)Ta{Hdl*2oD?4@GAvUVSPj(_lMy7!Ln+eE69;?&J()~OT>;H3}_zeUnaT#KGM zKFyW#wchJ3^zGV3GFr9urSX68oyu#llj6P0Dk~ZnQ(T@^jJG$ubB8DH_jQ^;3SnGQ zT1>_y%f-W0x;qK9wt%9lnZr~{-VcyuaOc4uM@GXt?dLeKR>4tDeR-co!nYlQ2~$Z3 zG6SG4hjC-w&u&egqx$%t#3LUE_3Js>PsgNAhoePRbDC6E1^Fa-uU=UfDzNAiRMC3> zmVO?cuANT;eI#s3nj$3ps3(ZMt$J43F)#H`kAj}%;Mk^vl2(@Puy}=dUG(f&v;kQX zcM_?E*8|>*Z-5e`qQ~%M`cZO`Dp_ItpFalVaP(lGJ0|5l5hO?SLIP~gKe1DN>+c=1FSdV(TP9s1{vqZ2v-P>NSp3vz1&p?BqbdXjBVEapUc|$jZi=H za_WZ6L#7*#d8Yj))hj zN}K!Pm1GqN(IwlQ+3r+V07nr@Bl0G^tYK;C09o;@wezqe%7|! zL+)N7Inu^Hzq0Up)E~V}G#^Np$@R2t0cAzryY2^5&RwT3!Za%Bwi)Jf*?IDNCi$4J4l=z~7{$j}?LNW9!%Hlmde z0@W)<2-TTHP}F^>2EH}t9&nFTm&o3}*ID1aC{8ZN6jb)^)t<}W2G5_4oPK{1)Juv| zN~iJGd8JS%PItZy(qL4sIhiVZpB=eP$V9*dl1-Z4?z#PcE$mq(=MmszFe`*#?!83v zY~<#PAM6<#aB#DF{uojGy)DNdB+5r{I~7M7^$R8<4u3^&V5-T29Om@Nj2sU1WVqWV zzC_P|Tv{Gk%0@f$h#5fQH};=bC$139$ViY{5*>i+I18m%-mDbi-2Y7e^w>~qoD?+% zh!to8`7)albnbS~LYFYGv>m!6MoSdGp%wt11OewGf)9vygn`higGIbw`8(rIjzSxw zOHjk$hD%=~-h<0X%0M3dgE1w*lID>nypWvHg`N&63w}#-$$pWvl;nVY)dVh*KcqsA z)Ijkeom9m7aj2UQEGHiQ=^W9kbOXUo{xFRAF!47c8Hfx~4dew#l&FRr7j+E%(E8?w zgkRUHE{0nSHFzsO6B2K}@(ht|P3%Uv0kR;-zQ=yKxyD$t;5B1SNJapBZm8?1m8rlZ F{|6_A8`J;* literal 0 HcmV?d00001 diff --git a/kubespray/contrib/terraform/aws/modules/iam/main.tf b/kubespray/contrib/terraform/aws/modules/iam/main.tf new file mode 100644 index 0000000..a35afc7 --- /dev/null +++ b/kubespray/contrib/terraform/aws/modules/iam/main.tf @@ -0,0 +1,141 @@ +#Add AWS Roles for Kubernetes + +resource "aws_iam_role" "kube_control_plane" { + name = "kubernetes-${var.aws_cluster_name}-master" + + assume_role_policy = < 0) ? (aws_instance.k8s-etcd.*.private_ip) : (aws_instance.k8s-master.*.private_ip))) +} + +output "aws_nlb_api_fqdn" { + value = "${module.aws-nlb.aws_nlb_api_fqdn}:${var.aws_nlb_api_port}" +} + +output "inventory" { + value = data.template_file.inventory.rendered +} + +output "default_tags" { + value = var.default_tags +} diff --git a/kubespray/contrib/terraform/aws/sample-inventory/cluster.tfvars b/kubespray/contrib/terraform/aws/sample-inventory/cluster.tfvars new file mode 100644 index 0000000..8aca219 --- /dev/null +++ b/kubespray/contrib/terraform/aws/sample-inventory/cluster.tfvars @@ -0,0 +1,59 @@ +#Global Vars +aws_cluster_name = "devtest" + +#VPC Vars +aws_vpc_cidr_block = "10.250.192.0/18" + +aws_cidr_subnets_private = ["10.250.192.0/20", "10.250.208.0/20"] + +aws_cidr_subnets_public = ["10.250.224.0/20", "10.250.240.0/20"] + +#Bastion Host +aws_bastion_num = 1 + +aws_bastion_size = "t2.medium" + +#Kubernetes Cluster + +aws_kube_master_num = 3 + +aws_kube_master_size = "t2.medium" + +aws_kube_master_disk_size = 50 + +aws_etcd_num = 3 + +aws_etcd_size = "t2.medium" + +aws_etcd_disk_size = 50 + +aws_kube_worker_num = 4 + +aws_kube_worker_size = "t2.medium" + +aws_kube_worker_disk_size = 50 + +#Settings AWS NLB + +aws_nlb_api_port = 6443 + +k8s_secure_api_port = 6443 + +default_tags = { + # Env = "devtest" # Product = "kubernetes" +} + +inventory_file = "../../../inventory/hosts" + +## Credentials +#AWS Access Key +AWS_ACCESS_KEY_ID = "" + +#AWS Secret Key +AWS_SECRET_ACCESS_KEY = "" + +#EC2 SSH Key Name +AWS_SSH_KEY_NAME = "" + +#AWS Region +AWS_DEFAULT_REGION = "eu-central-1" diff --git a/kubespray/contrib/terraform/aws/sample-inventory/group_vars b/kubespray/contrib/terraform/aws/sample-inventory/group_vars new file mode 120000 index 0000000..3735958 --- /dev/null +++ b/kubespray/contrib/terraform/aws/sample-inventory/group_vars @@ -0,0 +1 @@ +../../../../inventory/sample/group_vars \ No newline at end of file diff --git a/kubespray/contrib/terraform/aws/templates/inventory.tpl b/kubespray/contrib/terraform/aws/templates/inventory.tpl new file mode 100644 index 0000000..10a3995 --- /dev/null +++ b/kubespray/contrib/terraform/aws/templates/inventory.tpl @@ -0,0 +1,27 @@ +[all] +${connection_strings_master} +${connection_strings_node} +${connection_strings_etcd} +${public_ip_address_bastion} + +[bastion] +${public_ip_address_bastion} + +[kube_control_plane] +${list_master} + +[kube_node] +${list_node} + +[etcd] +${list_etcd} + +[calico_rr] + +[k8s_cluster:children] +kube_node +kube_control_plane +calico_rr + +[k8s_cluster:vars] +${nlb_api_fqdn} diff --git a/kubespray/contrib/terraform/aws/terraform.tfvars b/kubespray/contrib/terraform/aws/terraform.tfvars new file mode 100644 index 0000000..693fa9b --- /dev/null +++ b/kubespray/contrib/terraform/aws/terraform.tfvars @@ -0,0 +1,43 @@ +#Global Vars +aws_cluster_name = "devtest" + +#VPC Vars +aws_vpc_cidr_block = "10.250.192.0/18" +aws_cidr_subnets_private = ["10.250.192.0/20", "10.250.208.0/20"] +aws_cidr_subnets_public = ["10.250.224.0/20", "10.250.240.0/20"] + +# single AZ deployment +#aws_cidr_subnets_private = ["10.250.192.0/20"] +#aws_cidr_subnets_public = ["10.250.224.0/20"] + +# 3+ AZ deployment +#aws_cidr_subnets_private = ["10.250.192.0/24","10.250.193.0/24","10.250.194.0/24","10.250.195.0/24"] +#aws_cidr_subnets_public = ["10.250.224.0/24","10.250.225.0/24","10.250.226.0/24","10.250.227.0/24"] + +#Bastion Host +aws_bastion_num = 1 +aws_bastion_size = "t3.small" + +#Kubernetes Cluster +aws_kube_master_num = 3 +aws_kube_master_size = "t3.medium" +aws_kube_master_disk_size = 50 + +aws_etcd_num = 0 +aws_etcd_size = "t3.medium" +aws_etcd_disk_size = 50 + +aws_kube_worker_num = 4 +aws_kube_worker_size = "t3.medium" +aws_kube_worker_disk_size = 50 + +#Settings AWS ELB +aws_nlb_api_port = 6443 +k8s_secure_api_port = 6443 + +default_tags = { + # Env = "devtest" + # Product = "kubernetes" +} + +inventory_file = "../../../inventory/hosts" diff --git a/kubespray/contrib/terraform/aws/terraform.tfvars.example b/kubespray/contrib/terraform/aws/terraform.tfvars.example new file mode 100644 index 0000000..584b6a2 --- /dev/null +++ b/kubespray/contrib/terraform/aws/terraform.tfvars.example @@ -0,0 +1,33 @@ +#Global Vars +aws_cluster_name = "devtest" + +#VPC Vars +aws_vpc_cidr_block = "10.250.192.0/18" +aws_cidr_subnets_private = ["10.250.192.0/20","10.250.208.0/20"] +aws_cidr_subnets_public = ["10.250.224.0/20","10.250.240.0/20"] +aws_avail_zones = ["eu-central-1a","eu-central-1b"] + +#Bastion Host +aws_bastion_num = 1 +aws_bastion_size = "t3.small" + +#Kubernetes Cluster +aws_kube_master_num = 3 +aws_kube_master_size = "t3.medium" +aws_kube_master_disk_size = 50 + +aws_etcd_num = 3 +aws_etcd_size = "t3.medium" +aws_etcd_disk_size = 50 + +aws_kube_worker_num = 4 +aws_kube_worker_size = "t3.medium" +aws_kube_worker_disk_size = 50 + +#Settings AWS ELB +aws_nlb_api_port = 6443 +k8s_secure_api_port = 6443 + +default_tags = { } + +inventory_file = "../../../inventory/hosts" diff --git a/kubespray/contrib/terraform/aws/variables.tf b/kubespray/contrib/terraform/aws/variables.tf new file mode 100644 index 0000000..479629e --- /dev/null +++ b/kubespray/contrib/terraform/aws/variables.tf @@ -0,0 +1,125 @@ +variable "AWS_ACCESS_KEY_ID" { + description = "AWS Access Key" +} + +variable "AWS_SECRET_ACCESS_KEY" { + description = "AWS Secret Key" +} + +variable "AWS_SSH_KEY_NAME" { + description = "Name of the SSH keypair to use in AWS." +} + +variable "AWS_DEFAULT_REGION" { + description = "AWS Region" +} + +//General Cluster Settings + +variable "aws_cluster_name" { + description = "Name of AWS Cluster" +} + +data "aws_ami" "distro" { + most_recent = true + + filter { + name = "name" + values = ["debian-10-amd64-*"] + } + + filter { + name = "virtualization-type" + values = ["hvm"] + } + + owners = ["136693071363"] # Debian-10 +} + +//AWS VPC Variables + +variable "aws_vpc_cidr_block" { + description = "CIDR Block for VPC" +} + +variable "aws_cidr_subnets_private" { + description = "CIDR Blocks for private subnets in Availability Zones" + type = list(string) +} + +variable "aws_cidr_subnets_public" { + description = "CIDR Blocks for public subnets in Availability Zones" + type = list(string) +} + +//AWS EC2 Settings + +variable "aws_bastion_size" { + description = "EC2 Instance Size of Bastion Host" +} + +/* +* AWS EC2 Settings +* The number should be divisable by the number of used +* AWS Availability Zones without an remainder. +*/ +variable "aws_bastion_num" { + description = "Number of Bastion Nodes" +} + +variable "aws_kube_master_num" { + description = "Number of Kubernetes Master Nodes" +} + +variable "aws_kube_master_disk_size" { + description = "Disk size for Kubernetes Master Nodes (in GiB)" +} + +variable "aws_kube_master_size" { + description = "Instance size of Kube Master Nodes" +} + +variable "aws_etcd_num" { + description = "Number of etcd Nodes" +} + +variable "aws_etcd_disk_size" { + description = "Disk size for etcd Nodes (in GiB)" +} + +variable "aws_etcd_size" { + description = "Instance size of etcd Nodes" +} + +variable "aws_kube_worker_num" { + description = "Number of Kubernetes Worker Nodes" +} + +variable "aws_kube_worker_disk_size" { + description = "Disk size for Kubernetes Worker Nodes (in GiB)" +} + +variable "aws_kube_worker_size" { + description = "Instance size of Kubernetes Worker Nodes" +} + +/* +* AWS NLB Settings +* +*/ +variable "aws_nlb_api_port" { + description = "Port for AWS NLB" +} + +variable "k8s_secure_api_port" { + description = "Secure Port of K8S API Server" +} + +variable "default_tags" { + description = "Default tags for all resources" + type = map(string) +} + +variable "inventory_file" { + description = "Where to store the generated inventory file" +} diff --git a/kubespray/contrib/terraform/equinix/README.md b/kubespray/contrib/terraform/equinix/README.md new file mode 100644 index 0000000..d1eb71f --- /dev/null +++ b/kubespray/contrib/terraform/equinix/README.md @@ -0,0 +1,246 @@ +# Kubernetes on Equinix Metal with Terraform + +Provision a Kubernetes cluster with [Terraform](https://www.terraform.io) on +[Equinix Metal](https://metal.equinix.com) ([formerly Packet](https://blog.equinix.com/blog/2020/10/06/equinix-metal-metal-and-more/)). + +## Status + +This will install a Kubernetes cluster on Equinix Metal. It should work in all locations and on most server types. + +## Approach + +The terraform configuration inspects variables found in +[variables.tf](variables.tf) to create resources in your Equinix Metal project. +There is a [python script](../terraform.py) that reads the generated`.tfstate` +file to generate a dynamic inventory that is consumed by [cluster.yml](../../../cluster.yml) +to actually install Kubernetes with Kubespray. + +### Kubernetes Nodes + +You can create many different kubernetes topologies by setting the number of +different classes of hosts. + +- Master nodes with etcd: `number_of_k8s_masters` variable +- Master nodes without etcd: `number_of_k8s_masters_no_etcd` variable +- Standalone etcd hosts: `number_of_etcd` variable +- Kubernetes worker nodes: `number_of_k8s_nodes` variable + +Note that the Ansible script will report an invalid configuration if you wind up +with an *even number* of etcd instances since that is not a valid configuration. This +restriction includes standalone etcd nodes that are deployed in a cluster along with +master nodes with etcd replicas. As an example, if you have three master nodes with +etcd replicas and three standalone etcd nodes, the script will fail since there are +now six total etcd replicas. + +## Requirements + +- [Install Terraform](https://www.terraform.io/intro/getting-started/install.html) +- [Install Ansible dependencies](/docs/ansible.md#installing-ansible) +- Account with Equinix Metal +- An SSH key pair + +## SSH Key Setup + +An SSH keypair is required so Ansible can access the newly provisioned nodes (Equinix Metal hosts). By default, the public SSH key defined in cluster.tfvars will be installed in authorized_key on the newly provisioned nodes (~/.ssh/id_rsa.pub). Terraform will upload this public key and then it will be distributed out to all the nodes. If you have already set this public key in Equinix Metal (i.e. via the portal), then set the public keyfile name in cluster.tfvars to blank to prevent the duplicate key from being uploaded which will cause an error. + +If you don't already have a keypair generated (~/.ssh/id_rsa and ~/.ssh/id_rsa.pub), then a new keypair can be generated with the command: + +```ShellSession +ssh-keygen -f ~/.ssh/id_rsa +``` + +## Terraform + +Terraform will be used to provision all of the Equinix Metal resources with base software as appropriate. + +### Configuration + +#### Inventory files + +Create an inventory directory for your cluster by copying the existing sample and linking the `hosts` script (used to build the inventory based on Terraform state): + +```ShellSession +cp -LRp contrib/terraform/equinix/sample-inventory inventory/$CLUSTER +cd inventory/$CLUSTER +ln -s ../../contrib/terraform/equinix/hosts +``` + +This will be the base for subsequent Terraform commands. + +#### Equinix Metal API access + +Your Equinix Metal API key must be available in the `METAL_AUTH_TOKEN` environment variable. +This key is typically stored outside of the code repo since it is considered secret. +If someone gets this key, they can startup/shutdown hosts in your project! + +For more information on how to generate an API key or find your project ID, please see +[Accounts Index](https://metal.equinix.com/developers/docs/accounts/). + +The Equinix Metal Project ID associated with the key will be set later in `cluster.tfvars`. + +For more information about the API, please see [Equinix Metal API](https://metal.equinix.com/developers/api/). + +For more information about terraform provider authentication, please see [the equinix provider documentation](https://registry.terraform.io/providers/equinix/equinix/latest/docs). + +Example: + +```ShellSession +export METAL_AUTH_TOKEN="Example-API-Token" +``` + +Note that to deploy several clusters within the same project you need to use [terraform workspace](https://www.terraform.io/docs/state/workspaces.html#using-workspaces). + +#### Cluster variables + +The construction of the cluster is driven by values found in +[variables.tf](variables.tf). + +For your cluster, edit `inventory/$CLUSTER/cluster.tfvars`. + +The `cluster_name` is used to set a tag on each server deployed as part of this cluster. +This helps when identifying which hosts are associated with each cluster. + +While the defaults in variables.tf will successfully deploy a cluster, it is recommended to set the following values: + +- cluster_name = the name of the inventory directory created above as $CLUSTER +- equinix_metal_project_id = the Equinix Metal Project ID associated with the Equinix Metal API token above + +#### Enable localhost access + +Kubespray will pull down a Kubernetes configuration file to access this cluster by enabling the +`kubeconfig_localhost: true` in the Kubespray configuration. + +Edit `inventory/$CLUSTER/group_vars/k8s_cluster/k8s_cluster.yml` and comment back in the following line and change from `false` to `true`: +`\# kubeconfig_localhost: false` +becomes: +`kubeconfig_localhost: true` + +Once the Kubespray playbooks are run, a Kubernetes configuration file will be written to the local host at `inventory/$CLUSTER/artifacts/admin.conf` + +#### Terraform state files + +In the cluster's inventory folder, the following files might be created (either by Terraform +or manually), to prevent you from pushing them accidentally they are in a +`.gitignore` file in the `contrib/terraform/equinix` directory : + +- `.terraform` +- `.tfvars` +- `.tfstate` +- `.tfstate.backup` +- `.lock.hcl` + +You can still add them manually if you want to. + +### Initialization + +Before Terraform can operate on your cluster you need to install the required +plugins. This is accomplished as follows: + +```ShellSession +cd inventory/$CLUSTER +terraform -chdir=../../contrib/terraform/metal init -var-file=cluster.tfvars +``` + +This should finish fairly quickly telling you Terraform has successfully initialized and loaded necessary modules. + +### Provisioning cluster + +You can apply the Terraform configuration to your cluster with the following command +issued from your cluster's inventory directory (`inventory/$CLUSTER`): + +```ShellSession +terraform -chdir=../../contrib/terraform/equinix apply -var-file=cluster.tfvars +export ANSIBLE_HOST_KEY_CHECKING=False +ansible-playbook -i hosts ../../cluster.yml +``` + +### Destroying cluster + +You can destroy your new cluster with the following command issued from the cluster's inventory directory: + +```ShellSession +terraform -chdir=../../contrib/terraform/equinix destroy -var-file=cluster.tfvars +``` + +If you've started the Ansible run, it may also be a good idea to do some manual cleanup: + +- Remove SSH keys from the destroyed cluster from your `~/.ssh/known_hosts` file +- Clean up any temporary cache files: `rm /tmp/$CLUSTER-*` + +### Debugging + +You can enable debugging output from Terraform by setting `TF_LOG` to `DEBUG` before running the Terraform command. + +## Ansible + +### Node access + +#### SSH + +Ensure your local ssh-agent is running and your ssh key has been added. This +step is required by the terraform provisioner: + +```ShellSession +eval $(ssh-agent -s) +ssh-add ~/.ssh/id_rsa +``` + +If you have deployed and destroyed a previous iteration of your cluster, you will need to clear out any stale keys from your SSH "known hosts" file ( `~/.ssh/known_hosts`). + +#### Test access + +Make sure you can connect to the hosts. Note that Flatcar Container Linux by Kinvolk will have a state `FAILED` due to Python not being present. This is okay, because Python will be installed during bootstrapping, so long as the hosts are not `UNREACHABLE`. + +```ShellSession +$ ansible -i inventory/$CLUSTER/hosts -m ping all +example-k8s_node-1 | SUCCESS => { + "changed": false, + "ping": "pong" +} +example-etcd-1 | SUCCESS => { + "changed": false, + "ping": "pong" +} +example-k8s-master-1 | SUCCESS => { + "changed": false, + "ping": "pong" +} +``` + +If it fails try to connect manually via SSH. It could be something as simple as a stale host key. + +### Deploy Kubernetes + +```ShellSession +ansible-playbook --become -i inventory/$CLUSTER/hosts cluster.yml +``` + +This will take some time as there are many tasks to run. + +## Kubernetes + +### Set up kubectl + +- [Install kubectl](https://kubernetes.io/docs/tasks/tools/install-kubectl/) on the localhost. +- Verify that Kubectl runs correctly + +```ShellSession +kubectl version +``` + +- Verify that the Kubernetes configuration file has been copied over + +```ShellSession +cat inventory/alpha/$CLUSTER/admin.conf +``` + +- Verify that all the nodes are running correctly. + +```ShellSession +kubectl version +kubectl --kubeconfig=inventory/$CLUSTER/artifacts/admin.conf get nodes +``` + +## What's next + +Try out your new Kubernetes cluster with the [Hello Kubernetes service](https://kubernetes.io/docs/tasks/access-application-cluster/service-access-application-cluster/). diff --git a/kubespray/contrib/terraform/equinix/hosts b/kubespray/contrib/terraform/equinix/hosts new file mode 120000 index 0000000..804b6fa --- /dev/null +++ b/kubespray/contrib/terraform/equinix/hosts @@ -0,0 +1 @@ +../terraform.py \ No newline at end of file diff --git a/kubespray/contrib/terraform/equinix/kubespray.tf b/kubespray/contrib/terraform/equinix/kubespray.tf new file mode 100644 index 0000000..d6bd166 --- /dev/null +++ b/kubespray/contrib/terraform/equinix/kubespray.tf @@ -0,0 +1,57 @@ +resource "equinix_metal_ssh_key" "k8s" { + count = var.public_key_path != "" ? 1 : 0 + name = "kubernetes-${var.cluster_name}" + public_key = chomp(file(var.public_key_path)) +} + +resource "equinix_metal_device" "k8s_master" { + depends_on = [equinix_metal_ssh_key.k8s] + + count = var.number_of_k8s_masters + hostname = "${var.cluster_name}-k8s-master-${count.index + 1}" + plan = var.plan_k8s_masters + metro = var.metro + operating_system = var.operating_system + billing_cycle = var.billing_cycle + project_id = var.equinix_metal_project_id + tags = ["cluster-${var.cluster_name}", "k8s_cluster", "kube_control_plane", "etcd", "kube_node"] +} + +resource "equinix_metal_device" "k8s_master_no_etcd" { + depends_on = [equinix_metal_ssh_key.k8s] + + count = var.number_of_k8s_masters_no_etcd + hostname = "${var.cluster_name}-k8s-master-${count.index + 1}" + plan = var.plan_k8s_masters_no_etcd + metro = var.metro + operating_system = var.operating_system + billing_cycle = var.billing_cycle + project_id = var.equinix_metal_project_id + tags = ["cluster-${var.cluster_name}", "k8s_cluster", "kube_control_plane"] +} + +resource "equinix_metal_device" "k8s_etcd" { + depends_on = [equinix_metal_ssh_key.k8s] + + count = var.number_of_etcd + hostname = "${var.cluster_name}-etcd-${count.index + 1}" + plan = var.plan_etcd + metro = var.metro + operating_system = var.operating_system + billing_cycle = var.billing_cycle + project_id = var.equinix_metal_project_id + tags = ["cluster-${var.cluster_name}", "etcd"] +} + +resource "equinix_metal_device" "k8s_node" { + depends_on = [equinix_metal_ssh_key.k8s] + + count = var.number_of_k8s_nodes + hostname = "${var.cluster_name}-k8s-node-${count.index + 1}" + plan = var.plan_k8s_nodes + metro = var.metro + operating_system = var.operating_system + billing_cycle = var.billing_cycle + project_id = var.equinix_metal_project_id + tags = ["cluster-${var.cluster_name}", "k8s_cluster", "kube_node"] +} diff --git a/kubespray/contrib/terraform/equinix/output.tf b/kubespray/contrib/terraform/equinix/output.tf new file mode 100644 index 0000000..f4ab63a --- /dev/null +++ b/kubespray/contrib/terraform/equinix/output.tf @@ -0,0 +1,15 @@ +output "k8s_masters" { + value = equinix_metal_device.k8s_master.*.access_public_ipv4 +} + +output "k8s_masters_no_etc" { + value = equinix_metal_device.k8s_master_no_etcd.*.access_public_ipv4 +} + +output "k8s_etcds" { + value = equinix_metal_device.k8s_etcd.*.access_public_ipv4 +} + +output "k8s_nodes" { + value = equinix_metal_device.k8s_node.*.access_public_ipv4 +} diff --git a/kubespray/contrib/terraform/equinix/provider.tf b/kubespray/contrib/terraform/equinix/provider.tf new file mode 100644 index 0000000..61c0aba --- /dev/null +++ b/kubespray/contrib/terraform/equinix/provider.tf @@ -0,0 +1,17 @@ +terraform { + required_version = ">= 1.0.0" + + provider_meta "equinix" { + module_name = "kubespray" + } + required_providers { + equinix = { + source = "equinix/equinix" + version = "~> 1.14" + } + } +} + +# Configure the Equinix Metal Provider +provider "equinix" { +} diff --git a/kubespray/contrib/terraform/equinix/sample-inventory/cluster.tfvars b/kubespray/contrib/terraform/equinix/sample-inventory/cluster.tfvars new file mode 100644 index 0000000..039f208 --- /dev/null +++ b/kubespray/contrib/terraform/equinix/sample-inventory/cluster.tfvars @@ -0,0 +1,35 @@ +# your Kubernetes cluster name here +cluster_name = "mycluster" + +# Your Equinix Metal project ID. See https://metal.equinix.com/developers/docs/accounts/ +equinix_metal_project_id = "Example-Project-Id" + +# The public SSH key to be uploaded into authorized_keys in bare metal Equinix Metal nodes provisioned +# leave this value blank if the public key is already setup in the Equinix Metal project +# Terraform will complain if the public key is setup in Equinix Metal +public_key_path = "~/.ssh/id_rsa.pub" + +# Equinix interconnected bare metal across our global metros. +metro = "da" + +# operating_system +operating_system = "ubuntu_22_04" + +# standalone etcds +number_of_etcd = 0 + +plan_etcd = "t1.small.x86" + +# masters +number_of_k8s_masters = 1 + +number_of_k8s_masters_no_etcd = 0 + +plan_k8s_masters = "t1.small.x86" + +plan_k8s_masters_no_etcd = "t1.small.x86" + +# nodes +number_of_k8s_nodes = 2 + +plan_k8s_nodes = "t1.small.x86" diff --git a/kubespray/contrib/terraform/equinix/sample-inventory/group_vars b/kubespray/contrib/terraform/equinix/sample-inventory/group_vars new file mode 120000 index 0000000..3735958 --- /dev/null +++ b/kubespray/contrib/terraform/equinix/sample-inventory/group_vars @@ -0,0 +1 @@ +../../../../inventory/sample/group_vars \ No newline at end of file diff --git a/kubespray/contrib/terraform/equinix/variables.tf b/kubespray/contrib/terraform/equinix/variables.tf new file mode 100644 index 0000000..9417388 --- /dev/null +++ b/kubespray/contrib/terraform/equinix/variables.tf @@ -0,0 +1,56 @@ +variable "cluster_name" { + default = "kubespray" +} + +variable "equinix_metal_project_id" { + description = "Your Equinix Metal project ID. See https://metal.equinix.com/developers/docs/accounts/" +} + +variable "operating_system" { + default = "ubuntu_22_04" +} + +variable "public_key_path" { + description = "The path of the ssh pub key" + default = "~/.ssh/id_rsa.pub" +} + +variable "billing_cycle" { + default = "hourly" +} + +variable "metro" { + default = "da" +} + +variable "plan_k8s_masters" { + default = "c3.small.x86" +} + +variable "plan_k8s_masters_no_etcd" { + default = "c3.small.x86" +} + +variable "plan_etcd" { + default = "c3.small.x86" +} + +variable "plan_k8s_nodes" { + default = "c3.medium.x86" +} + +variable "number_of_k8s_masters" { + default = 1 +} + +variable "number_of_k8s_masters_no_etcd" { + default = 0 +} + +variable "number_of_etcd" { + default = 0 +} + +variable "number_of_k8s_nodes" { + default = 1 +} diff --git a/kubespray/contrib/terraform/exoscale/README.md b/kubespray/contrib/terraform/exoscale/README.md new file mode 100644 index 0000000..be451cc --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/README.md @@ -0,0 +1,152 @@ +# Kubernetes on Exoscale with Terraform + +Provision a Kubernetes cluster on [Exoscale](https://www.exoscale.com/) using Terraform and Kubespray + +## Overview + +The setup looks like following + +```text + Kubernetes cluster + +-----------------------+ ++---------------+ | +--------------+ | +| | | | +--------------+ | +| API server LB +---------> | | | | +| | | | | Master/etcd | | ++---------------+ | | | node(s) | | + | +-+ | | + | +--------------+ | + | ^ | + | | | + | v | ++---------------+ | +--------------+ | +| | | | +--------------+ | +| Ingress LB +---------> | | | | +| | | | | Worker | | ++---------------+ | | | node(s) | | + | +-+ | | + | +--------------+ | + +-----------------------+ +``` + +## Requirements + +* Terraform 0.13.0 or newer (0.12 also works if you modify the provider block to include version and remove all `versions.tf` files) + +## Quickstart + +NOTE: *Assumes you are at the root of the kubespray repo* + +Copy the sample inventory for your cluster and copy the default terraform variables. + +```bash +CLUSTER=my-exoscale-cluster +cp -r inventory/sample inventory/$CLUSTER +cp contrib/terraform/exoscale/default.tfvars inventory/$CLUSTER/ +cd inventory/$CLUSTER +``` + +Edit `default.tfvars` to match your setup. You MUST, at the very least, change `ssh_public_keys`. + +```bash +# Ensure $EDITOR points to your favorite editor, e.g., vim, emacs, VS Code, etc. +$EDITOR default.tfvars +``` + +For authentication you can use the credentials file `~/.cloudstack.ini` or `./cloudstack.ini`. +The file should look like something like this: + +```ini +[cloudstack] +key = +secret = +``` + +Follow the [Exoscale IAM Quick-start](https://community.exoscale.com/documentation/iam/quick-start/) to learn how to generate API keys. + +### Encrypted credentials + +To have the credentials encrypted at rest, you can use [sops](https://github.com/mozilla/sops) and only decrypt the credentials at runtime. + +```bash +cat << EOF > cloudstack.ini +[cloudstack] +key = +secret = +EOF +sops --encrypt --in-place --pgp cloudstack.ini +sops cloudstack.ini +``` + +Run terraform to create the infrastructure + +```bash +terraform init ../../contrib/terraform/exoscale +terraform apply -var-file default.tfvars ../../contrib/terraform/exoscale +``` + +If your cloudstack credentials file is encrypted using sops, run the following: + +```bash +terraform init ../../contrib/terraform/exoscale +sops exec-file -no-fifo cloudstack.ini 'CLOUDSTACK_CONFIG={} terraform apply -var-file default.tfvars ../../contrib/terraform/exoscale' +``` + +You should now have a inventory file named `inventory.ini` that you can use with kubespray. +You can now copy your inventory file and use it with kubespray to set up a cluster. +You can type `terraform output` to find out the IP addresses of the nodes, as well as control-plane and data-plane load-balancer. + +It is a good idea to check that you have basic SSH connectivity to the nodes. You can do that by: + +```bash +ansible -i inventory.ini -m ping all +``` + +Example to use this with the default sample inventory: + +```bash +ansible-playbook -i inventory.ini ../../cluster.yml -b -v +``` + +## Teardown + +The Kubernetes cluster cannot create any load-balancers or disks, hence, teardown is as simple as Terraform destroy: + +```bash +terraform destroy -var-file default.tfvars ../../contrib/terraform/exoscale +``` + +## Variables + +### Required + +* `ssh_public_keys`: List of public SSH keys to install on all machines +* `zone`: The zone where to run the cluster +* `machines`: Machines to provision. Key of this object will be used as the name of the machine + * `node_type`: The role of this node *(master|worker)* + * `size`: The size to use + * `boot_disk`: The boot disk to use + * `image_name`: Name of the image + * `root_partition_size`: Size *(in GB)* for the root partition + * `ceph_partition_size`: Size *(in GB)* for the partition for rook to use as ceph storage. *(Set to 0 to disable)* + * `node_local_partition_size`: Size *(in GB)* for the partition for node-local-storage. *(Set to 0 to disable)* +* `ssh_whitelist`: List of IP ranges (CIDR) that will be allowed to ssh to the nodes +* `api_server_whitelist`: List of IP ranges (CIDR) that will be allowed to connect to the API server +* `nodeport_whitelist`: List of IP ranges (CIDR) that will be allowed to connect to the kubernetes nodes on port 30000-32767 (kubernetes nodeports) + +### Optional + +* `prefix`: Prefix to use for all resources, required to be unique for all clusters in the same project *(Defaults to `default`)* + +An example variables file can be found `default.tfvars` + +## Known limitations + +### Only single disk + +Since Exoscale doesn't support additional disks to be mounted onto an instance, this script has the ability to create partitions for [Rook](https://rook.io/) and [node-local-storage](https://kubernetes.io/docs/concepts/storage/volumes/#local). + +### No Kubernetes API + +The current solution doesn't use the [Exoscale Kubernetes cloud controller](https://github.com/exoscale/exoscale-cloud-controller-manager). +This means that we need to set up a HTTP(S) loadbalancer in front of all workers and set the Ingress controller to DaemonSet mode. diff --git a/kubespray/contrib/terraform/exoscale/default.tfvars b/kubespray/contrib/terraform/exoscale/default.tfvars new file mode 100644 index 0000000..2bcbef5 --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/default.tfvars @@ -0,0 +1,65 @@ +prefix = "default" +zone = "ch-gva-2" + +inventory_file = "inventory.ini" + +ssh_public_keys = [ + # Put your public SSH key here + "ssh-rsa I-did-not-read-the-docs", + "ssh-rsa I-did-not-read-the-docs 2", +] + +machines = { + "master-0" : { + "node_type" : "master", + "size" : "Medium", + "boot_disk" : { + "image_name" : "Linux Ubuntu 20.04 LTS 64-bit", + "root_partition_size" : 50, + "node_local_partition_size" : 0, + "ceph_partition_size" : 0 + } + }, + "worker-0" : { + "node_type" : "worker", + "size" : "Large", + "boot_disk" : { + "image_name" : "Linux Ubuntu 20.04 LTS 64-bit", + "root_partition_size" : 50, + "node_local_partition_size" : 0, + "ceph_partition_size" : 0 + } + }, + "worker-1" : { + "node_type" : "worker", + "size" : "Large", + "boot_disk" : { + "image_name" : "Linux Ubuntu 20.04 LTS 64-bit", + "root_partition_size" : 50, + "node_local_partition_size" : 0, + "ceph_partition_size" : 0 + } + }, + "worker-2" : { + "node_type" : "worker", + "size" : "Large", + "boot_disk" : { + "image_name" : "Linux Ubuntu 20.04 LTS 64-bit", + "root_partition_size" : 50, + "node_local_partition_size" : 0, + "ceph_partition_size" : 0 + } + } +} + +nodeport_whitelist = [ + "0.0.0.0/0" +] + +ssh_whitelist = [ + "0.0.0.0/0" +] + +api_server_whitelist = [ + "0.0.0.0/0" +] diff --git a/kubespray/contrib/terraform/exoscale/main.tf b/kubespray/contrib/terraform/exoscale/main.tf new file mode 100644 index 0000000..eb9fcab --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/main.tf @@ -0,0 +1,49 @@ +provider "exoscale" {} + +module "kubernetes" { + source = "./modules/kubernetes-cluster" + + prefix = var.prefix + zone = var.zone + machines = var.machines + + ssh_public_keys = var.ssh_public_keys + + ssh_whitelist = var.ssh_whitelist + api_server_whitelist = var.api_server_whitelist + nodeport_whitelist = var.nodeport_whitelist +} + +# +# Generate ansible inventory +# + +data "template_file" "inventory" { + template = file("${path.module}/templates/inventory.tpl") + + vars = { + connection_strings_master = join("\n", formatlist("%s ansible_user=ubuntu ansible_host=%s ip=%s etcd_member_name=etcd%d", + keys(module.kubernetes.master_ip_addresses), + values(module.kubernetes.master_ip_addresses).*.public_ip, + values(module.kubernetes.master_ip_addresses).*.private_ip, + range(1, length(module.kubernetes.master_ip_addresses) + 1))) + connection_strings_worker = join("\n", formatlist("%s ansible_user=ubuntu ansible_host=%s ip=%s", + keys(module.kubernetes.worker_ip_addresses), + values(module.kubernetes.worker_ip_addresses).*.public_ip, + values(module.kubernetes.worker_ip_addresses).*.private_ip)) + + list_master = join("\n", keys(module.kubernetes.master_ip_addresses)) + list_worker = join("\n", keys(module.kubernetes.worker_ip_addresses)) + api_lb_ip_address = module.kubernetes.control_plane_lb_ip_address + } +} + +resource "null_resource" "inventories" { + provisioner "local-exec" { + command = "echo '${data.template_file.inventory.rendered}' > ${var.inventory_file}" + } + + triggers = { + template = data.template_file.inventory.rendered + } +} diff --git a/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/main.tf b/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/main.tf new file mode 100644 index 0000000..3171b00 --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/main.tf @@ -0,0 +1,193 @@ +data "exoscale_compute_template" "os_image" { + for_each = var.machines + + zone = var.zone + name = each.value.boot_disk.image_name +} + +data "exoscale_compute" "master_nodes" { + for_each = exoscale_compute.master + + id = each.value.id + + # Since private IP address is not assigned until the nics are created we need this + depends_on = [exoscale_nic.master_private_network_nic] +} + +data "exoscale_compute" "worker_nodes" { + for_each = exoscale_compute.worker + + id = each.value.id + + # Since private IP address is not assigned until the nics are created we need this + depends_on = [exoscale_nic.worker_private_network_nic] +} + +resource "exoscale_network" "private_network" { + zone = var.zone + name = "${var.prefix}-network" + + start_ip = cidrhost(var.private_network_cidr, 1) + # cidr -1 = Broadcast address + # cidr -2 = DHCP server address (exoscale specific) + end_ip = cidrhost(var.private_network_cidr, -3) + netmask = cidrnetmask(var.private_network_cidr) +} + +resource "exoscale_compute" "master" { + for_each = { + for name, machine in var.machines : + name => machine + if machine.node_type == "master" + } + + display_name = "${var.prefix}-${each.key}" + template_id = data.exoscale_compute_template.os_image[each.key].id + size = each.value.size + disk_size = each.value.boot_disk.root_partition_size + each.value.boot_disk.node_local_partition_size + each.value.boot_disk.ceph_partition_size + state = "Running" + zone = var.zone + security_groups = [exoscale_security_group.master_sg.name] + + user_data = templatefile( + "${path.module}/templates/cloud-init.tmpl", + { + eip_ip_address = exoscale_ipaddress.ingress_controller_lb.ip_address + node_local_partition_size = each.value.boot_disk.node_local_partition_size + ceph_partition_size = each.value.boot_disk.ceph_partition_size + root_partition_size = each.value.boot_disk.root_partition_size + node_type = "master" + ssh_public_keys = var.ssh_public_keys + } + ) +} + +resource "exoscale_compute" "worker" { + for_each = { + for name, machine in var.machines : + name => machine + if machine.node_type == "worker" + } + + display_name = "${var.prefix}-${each.key}" + template_id = data.exoscale_compute_template.os_image[each.key].id + size = each.value.size + disk_size = each.value.boot_disk.root_partition_size + each.value.boot_disk.node_local_partition_size + each.value.boot_disk.ceph_partition_size + state = "Running" + zone = var.zone + security_groups = [exoscale_security_group.worker_sg.name] + + user_data = templatefile( + "${path.module}/templates/cloud-init.tmpl", + { + eip_ip_address = exoscale_ipaddress.ingress_controller_lb.ip_address + node_local_partition_size = each.value.boot_disk.node_local_partition_size + ceph_partition_size = each.value.boot_disk.ceph_partition_size + root_partition_size = each.value.boot_disk.root_partition_size + node_type = "worker" + ssh_public_keys = var.ssh_public_keys + } + ) +} + +resource "exoscale_nic" "master_private_network_nic" { + for_each = exoscale_compute.master + + compute_id = each.value.id + network_id = exoscale_network.private_network.id +} + +resource "exoscale_nic" "worker_private_network_nic" { + for_each = exoscale_compute.worker + + compute_id = each.value.id + network_id = exoscale_network.private_network.id +} + +resource "exoscale_security_group" "master_sg" { + name = "${var.prefix}-master-sg" + description = "Security group for Kubernetes masters" +} + +resource "exoscale_security_group_rules" "master_sg_rules" { + security_group_id = exoscale_security_group.master_sg.id + + # SSH + ingress { + protocol = "TCP" + cidr_list = var.ssh_whitelist + ports = ["22"] + } + + # Kubernetes API + ingress { + protocol = "TCP" + cidr_list = var.api_server_whitelist + ports = ["6443"] + } +} + +resource "exoscale_security_group" "worker_sg" { + name = "${var.prefix}-worker-sg" + description = "security group for kubernetes worker nodes" +} + +resource "exoscale_security_group_rules" "worker_sg_rules" { + security_group_id = exoscale_security_group.worker_sg.id + + # SSH + ingress { + protocol = "TCP" + cidr_list = var.ssh_whitelist + ports = ["22"] + } + + # HTTP(S) + ingress { + protocol = "TCP" + cidr_list = ["0.0.0.0/0"] + ports = ["80", "443"] + } + + # Kubernetes Nodeport + ingress { + protocol = "TCP" + cidr_list = var.nodeport_whitelist + ports = ["30000-32767"] + } +} + +resource "exoscale_ipaddress" "ingress_controller_lb" { + zone = var.zone + healthcheck_mode = "http" + healthcheck_port = 80 + healthcheck_path = "/healthz" + healthcheck_interval = 10 + healthcheck_timeout = 2 + healthcheck_strikes_ok = 2 + healthcheck_strikes_fail = 3 +} + +resource "exoscale_secondary_ipaddress" "ingress_controller_lb" { + for_each = exoscale_compute.worker + + compute_id = each.value.id + ip_address = exoscale_ipaddress.ingress_controller_lb.ip_address +} + +resource "exoscale_ipaddress" "control_plane_lb" { + zone = var.zone + healthcheck_mode = "tcp" + healthcheck_port = 6443 + healthcheck_interval = 10 + healthcheck_timeout = 2 + healthcheck_strikes_ok = 2 + healthcheck_strikes_fail = 3 +} + +resource "exoscale_secondary_ipaddress" "control_plane_lb" { + for_each = exoscale_compute.master + + compute_id = each.value.id + ip_address = exoscale_ipaddress.control_plane_lb.ip_address +} diff --git a/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/output.tf b/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/output.tf new file mode 100644 index 0000000..bb80b5b --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/output.tf @@ -0,0 +1,31 @@ +output "master_ip_addresses" { + value = { + for key, instance in exoscale_compute.master : + instance.name => { + "private_ip" = contains(keys(data.exoscale_compute.master_nodes), key) ? data.exoscale_compute.master_nodes[key].private_network_ip_addresses[0] : "" + "public_ip" = exoscale_compute.master[key].ip_address + } + } +} + +output "worker_ip_addresses" { + value = { + for key, instance in exoscale_compute.worker : + instance.name => { + "private_ip" = contains(keys(data.exoscale_compute.worker_nodes), key) ? data.exoscale_compute.worker_nodes[key].private_network_ip_addresses[0] : "" + "public_ip" = exoscale_compute.worker[key].ip_address + } + } +} + +output "cluster_private_network_cidr" { + value = var.private_network_cidr +} + +output "ingress_controller_lb_ip_address" { + value = exoscale_ipaddress.ingress_controller_lb.ip_address +} + +output "control_plane_lb_ip_address" { + value = exoscale_ipaddress.control_plane_lb.ip_address +} diff --git a/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/templates/cloud-init.tmpl b/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/templates/cloud-init.tmpl new file mode 100644 index 0000000..a81b8e3 --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/templates/cloud-init.tmpl @@ -0,0 +1,52 @@ +#cloud-config +%{ if ceph_partition_size > 0 || node_local_partition_size > 0} +bootcmd: +- [ cloud-init-per, once, move-second-header, sgdisk, --move-second-header, /dev/vda ] +%{ if node_local_partition_size > 0 } + # Create partition for node local storage +- [ cloud-init-per, once, create-node-local-part, parted, --script, /dev/vda, 'mkpart extended ext4 ${root_partition_size}GB %{ if ceph_partition_size == 0 }-1%{ else }${root_partition_size + node_local_partition_size}GB%{ endif }' ] +- [ cloud-init-per, once, create-fs-node-local-part, mkfs.ext4, /dev/vda2 ] +%{ endif } +%{ if ceph_partition_size > 0 } + # Create partition for rook to use for ceph +- [ cloud-init-per, once, create-ceph-part, parted, --script, /dev/vda, 'mkpart extended ${root_partition_size + node_local_partition_size}GB -1' ] +%{ endif } +%{ endif } + +ssh_authorized_keys: +%{ for ssh_public_key in ssh_public_keys ~} + - ${ssh_public_key} +%{ endfor ~} + +write_files: + - path: /etc/netplan/eth1.yaml + content: | + network: + version: 2 + ethernets: + eth1: + dhcp4: true +%{ if node_type == "worker" } + # TODO: When a VM is seen as healthy and is added to the EIP loadbalancer + # pool it no longer can send traffic back to itself via the EIP IP + # address. + # Remove this if it ever gets solved. + - path: /etc/netplan/20-eip-fix.yaml + content: | + network: + version: 2 + ethernets: + "lo:0": + match: + name: lo + dhcp4: false + addresses: + - ${eip_ip_address}/32 +%{ endif } +runcmd: + - netplan apply +%{ if node_local_partition_size > 0 } + - mkdir -p /mnt/disks/node-local-storage + - chown nobody:nogroup /mnt/disks/node-local-storage + - mount /dev/vda2 /mnt/disks/node-local-storage +%{ endif } diff --git a/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/variables.tf b/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/variables.tf new file mode 100644 index 0000000..c466abf --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/variables.tf @@ -0,0 +1,42 @@ +variable "zone" { + type = string + # This is currently the only zone that is supposed to be supporting + # so called "managed private networks". + # See: https://www.exoscale.com/syslog/introducing-managed-private-networks + default = "ch-gva-2" +} + +variable "prefix" {} + +variable "machines" { + type = map(object({ + node_type = string + size = string + boot_disk = object({ + image_name = string + root_partition_size = number + ceph_partition_size = number + node_local_partition_size = number + }) + })) +} + +variable "ssh_public_keys" { + type = list(string) +} + +variable "ssh_whitelist" { + type = list(string) +} + +variable "api_server_whitelist" { + type = list(string) +} + +variable "nodeport_whitelist" { + type = list(string) +} + +variable "private_network_cidr" { + default = "172.0.10.0/24" +} diff --git a/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/versions.tf b/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/versions.tf new file mode 100644 index 0000000..6f60994 --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/modules/kubernetes-cluster/versions.tf @@ -0,0 +1,9 @@ +terraform { + required_providers { + exoscale = { + source = "exoscale/exoscale" + version = ">= 0.21" + } + } + required_version = ">= 0.13" +} diff --git a/kubespray/contrib/terraform/exoscale/output.tf b/kubespray/contrib/terraform/exoscale/output.tf new file mode 100644 index 0000000..09bf7fa --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/output.tf @@ -0,0 +1,15 @@ +output "master_ips" { + value = module.kubernetes.master_ip_addresses +} + +output "worker_ips" { + value = module.kubernetes.worker_ip_addresses +} + +output "ingress_controller_lb_ip_address" { + value = module.kubernetes.ingress_controller_lb_ip_address +} + +output "control_plane_lb_ip_address" { + value = module.kubernetes.control_plane_lb_ip_address +} diff --git a/kubespray/contrib/terraform/exoscale/sample-inventory/cluster.tfvars b/kubespray/contrib/terraform/exoscale/sample-inventory/cluster.tfvars new file mode 100644 index 0000000..f615241 --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/sample-inventory/cluster.tfvars @@ -0,0 +1,65 @@ +prefix = "default" +zone = "ch-gva-2" + +inventory_file = "inventory.ini" + +ssh_public_keys = [ + # Put your public SSH key here + "ssh-rsa I-did-not-read-the-docs", + "ssh-rsa I-did-not-read-the-docs 2", +] + +machines = { + "master-0" : { + "node_type" : "master", + "size" : "Small", + "boot_disk" : { + "image_name" : "Linux Ubuntu 20.04 LTS 64-bit", + "root_partition_size" : 50, + "node_local_partition_size" : 0, + "ceph_partition_size" : 0 + } + }, + "worker-0" : { + "node_type" : "worker", + "size" : "Large", + "boot_disk" : { + "image_name" : "Linux Ubuntu 20.04 LTS 64-bit", + "root_partition_size" : 50, + "node_local_partition_size" : 0, + "ceph_partition_size" : 0 + } + }, + "worker-1" : { + "node_type" : "worker", + "size" : "Large", + "boot_disk" : { + "image_name" : "Linux Ubuntu 20.04 LTS 64-bit", + "root_partition_size" : 50, + "node_local_partition_size" : 0, + "ceph_partition_size" : 0 + } + }, + "worker-2" : { + "node_type" : "worker", + "size" : "Large", + "boot_disk" : { + "image_name" : "Linux Ubuntu 20.04 LTS 64-bit", + "root_partition_size" : 50, + "node_local_partition_size" : 0, + "ceph_partition_size" : 0 + } + } +} + +nodeport_whitelist = [ + "0.0.0.0/0" +] + +ssh_whitelist = [ + "0.0.0.0/0" +] + +api_server_whitelist = [ + "0.0.0.0/0" +] diff --git a/kubespray/contrib/terraform/exoscale/sample-inventory/group_vars b/kubespray/contrib/terraform/exoscale/sample-inventory/group_vars new file mode 120000 index 0000000..3735958 --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/sample-inventory/group_vars @@ -0,0 +1 @@ +../../../../inventory/sample/group_vars \ No newline at end of file diff --git a/kubespray/contrib/terraform/exoscale/templates/inventory.tpl b/kubespray/contrib/terraform/exoscale/templates/inventory.tpl new file mode 100644 index 0000000..85ed192 --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/templates/inventory.tpl @@ -0,0 +1,19 @@ +[all] +${connection_strings_master} +${connection_strings_worker} + +[kube_control_plane] +${list_master} + +[kube_control_plane:vars] +supplementary_addresses_in_ssl_keys = [ "${api_lb_ip_address}" ] + +[etcd] +${list_master} + +[kube_node] +${list_worker} + +[k8s_cluster:children] +kube_control_plane +kube_node diff --git a/kubespray/contrib/terraform/exoscale/variables.tf b/kubespray/contrib/terraform/exoscale/variables.tf new file mode 100644 index 0000000..14f8455 --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/variables.tf @@ -0,0 +1,46 @@ +variable "zone" { + description = "The zone where to run the cluster" +} + +variable "prefix" { + description = "Prefix for resource names" + default = "default" +} + +variable "machines" { + description = "Cluster machines" + type = map(object({ + node_type = string + size = string + boot_disk = object({ + image_name = string + root_partition_size = number + ceph_partition_size = number + node_local_partition_size = number + }) + })) +} + +variable "ssh_public_keys" { + description = "List of public SSH keys which are injected into the VMs." + type = list(string) +} + +variable "ssh_whitelist" { + description = "List of IP ranges (CIDR) to whitelist for ssh" + type = list(string) +} + +variable "api_server_whitelist" { + description = "List of IP ranges (CIDR) to whitelist for kubernetes api server" + type = list(string) +} + +variable "nodeport_whitelist" { + description = "List of IP ranges (CIDR) to whitelist for kubernetes nodeports" + type = list(string) +} + +variable "inventory_file" { + description = "Where to store the generated inventory file" +} diff --git a/kubespray/contrib/terraform/exoscale/versions.tf b/kubespray/contrib/terraform/exoscale/versions.tf new file mode 100644 index 0000000..0333b41 --- /dev/null +++ b/kubespray/contrib/terraform/exoscale/versions.tf @@ -0,0 +1,15 @@ +terraform { + required_providers { + exoscale = { + source = "exoscale/exoscale" + version = ">= 0.21" + } + null = { + source = "hashicorp/null" + } + template = { + source = "hashicorp/template" + } + } + required_version = ">= 0.13" +} diff --git a/kubespray/contrib/terraform/gcp/README.md b/kubespray/contrib/terraform/gcp/README.md new file mode 100644 index 0000000..01e5299 --- /dev/null +++ b/kubespray/contrib/terraform/gcp/README.md @@ -0,0 +1,104 @@ +# Kubernetes on GCP with Terraform + +Provision a Kubernetes cluster on GCP using Terraform and Kubespray + +## Overview + +The setup looks like following + +```text + Kubernetes cluster + +-----------------------+ ++---------------+ | +--------------+ | +| | | | +--------------+ | +| API server LB +---------> | | | | +| | | | | Master/etcd | | ++---------------+ | | | node(s) | | + | +-+ | | + | +--------------+ | + | ^ | + | | | + | v | ++---------------+ | +--------------+ | +| | | | +--------------+ | +| Ingress LB +---------> | | | | +| | | | | Worker | | ++---------------+ | | | node(s) | | + | +-+ | | + | +--------------+ | + +-----------------------+ +``` + +## Requirements + +* Terraform 0.12.0 or newer + +## Quickstart + +To get a cluster up and running you'll need a JSON keyfile. +Set the path to the file in the `tfvars.json` file and run the following: + +```bash +terraform apply -var-file tfvars.json -state dev-cluster.tfstate -var gcp_project_id= -var keyfile_location= +``` + +To generate kubespray inventory based on the terraform state file you can run the following: + +```bash +./generate-inventory.sh dev-cluster.tfstate > inventory.ini +``` + +You should now have a inventory file named `inventory.ini` that you can use with kubespray, e.g. + +```bash +ansible-playbook -i contrib/terraform/gcs/inventory.ini cluster.yml -b -v +``` + +## Variables + +### Required + +* `keyfile_location`: Location to the keyfile to use as credentials for the google terraform provider +* `gcp_project_id`: ID of the GCP project to deploy the cluster in +* `ssh_pub_key`: Path to public ssh key to use for all machines +* `region`: The region where to run the cluster +* `machines`: Machines to provision. Key of this object will be used as the name of the machine + * `node_type`: The role of this node *(master|worker)* + * `size`: The size to use + * `zone`: The zone the machine should run in + * `additional_disks`: Extra disks to add to the machine. Key of this object will be used as the disk name + * `size`: Size of the disk (in GB) + * `boot_disk`: The boot disk to use + * `image_name`: Name of the image + * `size`: Size of the boot disk (in GB) +* `ssh_whitelist`: List of IP ranges (CIDR) that will be allowed to ssh to the nodes +* `api_server_whitelist`: List of IP ranges (CIDR) that will be allowed to connect to the API server +* `nodeport_whitelist`: List of IP ranges (CIDR) that will be allowed to connect to the kubernetes nodes on port 30000-32767 (kubernetes nodeports) +* `ingress_whitelist`: List of IP ranges (CIDR) that will be allowed to connect to ingress on ports 80 and 443 +* `extra_ingress_firewalls`: Additional ingress firewall rules. Key will be used as the name of the rule + * `source_ranges`: List of IP ranges (CIDR). Example: `["8.8.8.8"]` + * `protocol`: Protocol. Example `"tcp"` + * `ports`: List of ports, as string. Example `["53"]` + * `target_tags`: List of target tag (either the machine name or `control-plane` or `worker`). Example: `["control-plane", "worker-0"]` + +### Optional + +* `prefix`: Prefix to use for all resources, required to be unique for all clusters in the same project *(Defaults to `default`)* +* `master_sa_email`: Service account email to use for the control plane nodes *(Defaults to `""`, auto generate one)* +* `master_sa_scopes`: Service account email to use for the control plane nodes *(Defaults to `["https://www.googleapis.com/auth/cloud-platform"]`)* +* `master_preemptible`: Enable [preemptible](https://cloud.google.com/compute/docs/instances/preemptible) + for the control plane nodes *(Defaults to `false`)* +* `master_additional_disk_type`: [Disk type](https://cloud.google.com/compute/docs/disks/#disk-types) + for extra disks added on the control plane nodes *(Defaults to `"pd-ssd"`)* +* `worker_sa_email`: Service account email to use for the worker nodes *(Defaults to `""`, auto generate one)* +* `worker_sa_scopes`: Service account email to use for the worker nodes *(Defaults to `["https://www.googleapis.com/auth/cloud-platform"]`)* +* `worker_preemptible`: Enable [preemptible](https://cloud.google.com/compute/docs/instances/preemptible) + for the worker nodes *(Defaults to `false`)* +* `worker_additional_disk_type`: [Disk type](https://cloud.google.com/compute/docs/disks/#disk-types) + for extra disks added on the worker nodes *(Defaults to `"pd-ssd"`)* + +An example variables file can be found `tfvars.json` + +## Known limitations + +This solution does not provide a solution to use a bastion host. Thus all the nodes must expose a public IP for kubespray to work. diff --git a/kubespray/contrib/terraform/gcp/generate-inventory.sh b/kubespray/contrib/terraform/gcp/generate-inventory.sh new file mode 100755 index 0000000..585a4f4 --- /dev/null +++ b/kubespray/contrib/terraform/gcp/generate-inventory.sh @@ -0,0 +1,76 @@ +#!/bin/bash + +# +# Generates a inventory file based on the terraform output. +# After provisioning a cluster, simply run this command and supply the terraform state file +# Default state file is terraform.tfstate +# + +set -e + +usage () { + echo "Usage: $0 " >&2 + exit 1 +} + +if [[ $# -ne 1 ]]; then + usage +fi + +TF_STATE_FILE=${1} + +if [[ ! -f "${TF_STATE_FILE}" ]]; then + echo "ERROR: state file ${TF_STATE_FILE} doesn't exist" >&2 + usage +fi + +TF_OUT=$(terraform output -state "${TF_STATE_FILE}" -json) + +MASTERS=$(jq -r '.master_ips.value | to_entries[]' <(echo "${TF_OUT}")) +WORKERS=$(jq -r '.worker_ips.value | to_entries[]' <(echo "${TF_OUT}")) +mapfile -t MASTER_NAMES < <(jq -r '.key' <(echo "${MASTERS}")) +mapfile -t WORKER_NAMES < <(jq -r '.key' <(echo "${WORKERS}")) + +API_LB=$(jq -r '.control_plane_lb_ip_address.value' <(echo "${TF_OUT}")) + +# Generate master hosts +i=1 +for name in "${MASTER_NAMES[@]}"; do + private_ip=$(jq -r '. | select( .key=='"\"${name}\""' ) | .value.private_ip' <(echo "${MASTERS}")) + public_ip=$(jq -r '. | select( .key=='"\"${name}\""' ) | .value.public_ip' <(echo "${MASTERS}")) + echo "${name} ansible_user=ubuntu ansible_host=${public_ip} ip=${private_ip} etcd_member_name=etcd${i}" + i=$(( i + 1 )) +done + +# Generate worker hosts +for name in "${WORKER_NAMES[@]}"; do + private_ip=$(jq -r '. | select( .key=='"\"${name}\""' ) | .value.private_ip' <(echo "${WORKERS}")) + public_ip=$(jq -r '. | select( .key=='"\"${name}\""' ) | .value.public_ip' <(echo "${WORKERS}")) + echo "${name} ansible_user=ubuntu ansible_host=${public_ip} ip=${private_ip}" +done + +echo "" +echo "[kube_control_plane]" +for name in "${MASTER_NAMES[@]}"; do + echo "${name}" +done + +echo "" +echo "[kube_control_plane:vars]" +echo "supplementary_addresses_in_ssl_keys = [ '${API_LB}' ]" # Add LB address to API server certificate +echo "" +echo "[etcd]" +for name in "${MASTER_NAMES[@]}"; do + echo "${name}" +done + +echo "" +echo "[kube_node]" +for name in "${WORKER_NAMES[@]}"; do + echo "${name}" +done + +echo "" +echo "[k8s_cluster:children]" +echo "kube_control_plane" +echo "kube_node" diff --git a/kubespray/contrib/terraform/gcp/main.tf b/kubespray/contrib/terraform/gcp/main.tf new file mode 100644 index 0000000..b0b91f5 --- /dev/null +++ b/kubespray/contrib/terraform/gcp/main.tf @@ -0,0 +1,39 @@ +terraform { + required_providers { + google = { + source = "hashicorp/google" + version = "~> 4.0" + } + } +} + +provider "google" { + credentials = file(var.keyfile_location) + region = var.region + project = var.gcp_project_id +} + +module "kubernetes" { + source = "./modules/kubernetes-cluster" + region = var.region + prefix = var.prefix + + machines = var.machines + ssh_pub_key = var.ssh_pub_key + + master_sa_email = var.master_sa_email + master_sa_scopes = var.master_sa_scopes + master_preemptible = var.master_preemptible + master_additional_disk_type = var.master_additional_disk_type + worker_sa_email = var.worker_sa_email + worker_sa_scopes = var.worker_sa_scopes + worker_preemptible = var.worker_preemptible + worker_additional_disk_type = var.worker_additional_disk_type + + ssh_whitelist = var.ssh_whitelist + api_server_whitelist = var.api_server_whitelist + nodeport_whitelist = var.nodeport_whitelist + ingress_whitelist = var.ingress_whitelist + + extra_ingress_firewalls = var.extra_ingress_firewalls +} diff --git a/kubespray/contrib/terraform/gcp/modules/kubernetes-cluster/main.tf b/kubespray/contrib/terraform/gcp/modules/kubernetes-cluster/main.tf new file mode 100644 index 0000000..a83b73b --- /dev/null +++ b/kubespray/contrib/terraform/gcp/modules/kubernetes-cluster/main.tf @@ -0,0 +1,421 @@ +################################################# +## +## General +## + +resource "google_compute_network" "main" { + name = "${var.prefix}-network" + + auto_create_subnetworks = false +} + +resource "google_compute_subnetwork" "main" { + name = "${var.prefix}-subnet" + network = google_compute_network.main.name + ip_cidr_range = var.private_network_cidr + region = var.region +} + +resource "google_compute_firewall" "deny_all" { + name = "${var.prefix}-default-firewall" + network = google_compute_network.main.name + + priority = 1000 + + source_ranges = ["0.0.0.0/0"] + + deny { + protocol = "all" + } +} + +resource "google_compute_firewall" "allow_internal" { + name = "${var.prefix}-internal-firewall" + network = google_compute_network.main.name + + priority = 500 + + source_ranges = [var.private_network_cidr] + + allow { + protocol = "all" + } +} + +resource "google_compute_firewall" "ssh" { + count = length(var.ssh_whitelist) > 0 ? 1 : 0 + + name = "${var.prefix}-ssh-firewall" + network = google_compute_network.main.name + + priority = 100 + + source_ranges = var.ssh_whitelist + + allow { + protocol = "tcp" + ports = ["22"] + } +} + +resource "google_compute_firewall" "api_server" { + count = length(var.api_server_whitelist) > 0 ? 1 : 0 + + name = "${var.prefix}-api-server-firewall" + network = google_compute_network.main.name + + priority = 100 + + source_ranges = var.api_server_whitelist + + allow { + protocol = "tcp" + ports = ["6443"] + } +} + +resource "google_compute_firewall" "nodeport" { + count = length(var.nodeport_whitelist) > 0 ? 1 : 0 + + name = "${var.prefix}-nodeport-firewall" + network = google_compute_network.main.name + + priority = 100 + + source_ranges = var.nodeport_whitelist + + allow { + protocol = "tcp" + ports = ["30000-32767"] + } +} + +resource "google_compute_firewall" "ingress_http" { + count = length(var.ingress_whitelist) > 0 ? 1 : 0 + + name = "${var.prefix}-http-ingress-firewall" + network = google_compute_network.main.name + + priority = 100 + + source_ranges = var.ingress_whitelist + + allow { + protocol = "tcp" + ports = ["80"] + } +} + +resource "google_compute_firewall" "ingress_https" { + count = length(var.ingress_whitelist) > 0 ? 1 : 0 + + name = "${var.prefix}-https-ingress-firewall" + network = google_compute_network.main.name + + priority = 100 + + source_ranges = var.ingress_whitelist + + allow { + protocol = "tcp" + ports = ["443"] + } +} + +################################################# +## +## Local variables +## + +locals { + master_target_list = [ + for name, machine in google_compute_instance.master : + "${machine.zone}/${machine.name}" + ] + + worker_target_list = [ + for name, machine in google_compute_instance.worker : + "${machine.zone}/${machine.name}" + ] + + master_disks = flatten([ + for machine_name, machine in var.machines : [ + for disk_name, disk in machine.additional_disks : { + "${machine_name}-${disk_name}" = { + "machine_name": machine_name, + "machine": machine, + "disk_size": disk.size, + "disk_name": disk_name + } + } + ] + if machine.node_type == "master" + ]) + + worker_disks = flatten([ + for machine_name, machine in var.machines : [ + for disk_name, disk in machine.additional_disks : { + "${machine_name}-${disk_name}" = { + "machine_name": machine_name, + "machine": machine, + "disk_size": disk.size, + "disk_name": disk_name + } + } + ] + if machine.node_type == "worker" + ]) +} + +################################################# +## +## Master +## + +resource "google_compute_address" "master" { + for_each = { + for name, machine in var.machines : + name => machine + if machine.node_type == "master" + } + + name = "${var.prefix}-${each.key}-pip" + address_type = "EXTERNAL" + region = var.region +} + +resource "google_compute_disk" "master" { + for_each = { + for item in local.master_disks : + keys(item)[0] => values(item)[0] + } + + name = "${var.prefix}-${each.key}" + type = var.master_additional_disk_type + zone = each.value.machine.zone + size = each.value.disk_size + + physical_block_size_bytes = 4096 +} + +resource "google_compute_attached_disk" "master" { + for_each = { + for item in local.master_disks : + keys(item)[0] => values(item)[0] + } + + disk = google_compute_disk.master[each.key].id + instance = google_compute_instance.master[each.value.machine_name].id +} + +resource "google_compute_instance" "master" { + for_each = { + for name, machine in var.machines : + name => machine + if machine.node_type == "master" + } + + name = "${var.prefix}-${each.key}" + machine_type = each.value.size + zone = each.value.zone + + tags = ["control-plane", "master", each.key] + + boot_disk { + initialize_params { + image = each.value.boot_disk.image_name + size = each.value.boot_disk.size + } + } + + network_interface { + subnetwork = google_compute_subnetwork.main.name + + access_config { + nat_ip = google_compute_address.master[each.key].address + } + } + + metadata = { + ssh-keys = "ubuntu:${trimspace(file(pathexpand(var.ssh_pub_key)))}" + } + + service_account { + email = var.master_sa_email + scopes = var.master_sa_scopes + } + + # Since we use google_compute_attached_disk we need to ignore this + lifecycle { + ignore_changes = [attached_disk] + } + + scheduling { + preemptible = var.master_preemptible + automatic_restart = !var.master_preemptible + } +} + +resource "google_compute_forwarding_rule" "master_lb" { + count = length(var.api_server_whitelist) > 0 ? 1 : 0 + + name = "${var.prefix}-master-lb-forward-rule" + + port_range = "6443" + + target = google_compute_target_pool.master_lb[count.index].id +} + +resource "google_compute_target_pool" "master_lb" { + count = length(var.api_server_whitelist) > 0 ? 1 : 0 + + name = "${var.prefix}-master-lb-pool" + instances = local.master_target_list +} + +################################################# +## +## Worker +## + +resource "google_compute_disk" "worker" { + for_each = { + for item in local.worker_disks : + keys(item)[0] => values(item)[0] + } + + name = "${var.prefix}-${each.key}" + type = var.worker_additional_disk_type + zone = each.value.machine.zone + size = each.value.disk_size + + physical_block_size_bytes = 4096 +} + +resource "google_compute_attached_disk" "worker" { + for_each = { + for item in local.worker_disks : + keys(item)[0] => values(item)[0] + } + + disk = google_compute_disk.worker[each.key].id + instance = google_compute_instance.worker[each.value.machine_name].id +} + +resource "google_compute_address" "worker" { + for_each = { + for name, machine in var.machines : + name => machine + if machine.node_type == "worker" + } + + name = "${var.prefix}-${each.key}-pip" + address_type = "EXTERNAL" + region = var.region +} + +resource "google_compute_instance" "worker" { + for_each = { + for name, machine in var.machines : + name => machine + if machine.node_type == "worker" + } + + name = "${var.prefix}-${each.key}" + machine_type = each.value.size + zone = each.value.zone + + tags = ["worker", each.key] + + boot_disk { + initialize_params { + image = each.value.boot_disk.image_name + size = each.value.boot_disk.size + } + } + + network_interface { + subnetwork = google_compute_subnetwork.main.name + + access_config { + nat_ip = google_compute_address.worker[each.key].address + } + } + + metadata = { + ssh-keys = "ubuntu:${trimspace(file(pathexpand(var.ssh_pub_key)))}" + } + + service_account { + email = var.worker_sa_email + scopes = var.worker_sa_scopes + } + + # Since we use google_compute_attached_disk we need to ignore this + lifecycle { + ignore_changes = [attached_disk] + } + + scheduling { + preemptible = var.worker_preemptible + automatic_restart = !var.worker_preemptible + } +} + +resource "google_compute_address" "worker_lb" { + count = length(var.ingress_whitelist) > 0 ? 1 : 0 + + name = "${var.prefix}-worker-lb-address" + address_type = "EXTERNAL" + region = var.region +} + +resource "google_compute_forwarding_rule" "worker_http_lb" { + count = length(var.ingress_whitelist) > 0 ? 1 : 0 + + name = "${var.prefix}-worker-http-lb-forward-rule" + + ip_address = google_compute_address.worker_lb[count.index].address + port_range = "80" + + target = google_compute_target_pool.worker_lb[count.index].id +} + +resource "google_compute_forwarding_rule" "worker_https_lb" { + count = length(var.ingress_whitelist) > 0 ? 1 : 0 + + name = "${var.prefix}-worker-https-lb-forward-rule" + + ip_address = google_compute_address.worker_lb[count.index].address + port_range = "443" + + target = google_compute_target_pool.worker_lb[count.index].id +} + +resource "google_compute_target_pool" "worker_lb" { + count = length(var.ingress_whitelist) > 0 ? 1 : 0 + + name = "${var.prefix}-worker-lb-pool" + instances = local.worker_target_list +} + +resource "google_compute_firewall" "extra_ingress_firewall" { + for_each = { + for name, firewall in var.extra_ingress_firewalls : + name => firewall + } + + name = "${var.prefix}-${each.key}-ingress" + network = google_compute_network.main.name + + priority = 100 + + source_ranges = each.value.source_ranges + + target_tags = each.value.target_tags + + allow { + protocol = each.value.protocol + ports = each.value.ports + } +} diff --git a/kubespray/contrib/terraform/gcp/modules/kubernetes-cluster/output.tf b/kubespray/contrib/terraform/gcp/modules/kubernetes-cluster/output.tf new file mode 100644 index 0000000..d0ffaa9 --- /dev/null +++ b/kubespray/contrib/terraform/gcp/modules/kubernetes-cluster/output.tf @@ -0,0 +1,27 @@ +output "master_ip_addresses" { + value = { + for key, instance in google_compute_instance.master : + instance.name => { + "private_ip" = instance.network_interface.0.network_ip + "public_ip" = instance.network_interface.0.access_config.0.nat_ip + } + } +} + +output "worker_ip_addresses" { + value = { + for key, instance in google_compute_instance.worker : + instance.name => { + "private_ip" = instance.network_interface.0.network_ip + "public_ip" = instance.network_interface.0.access_config.0.nat_ip + } + } +} + +output "ingress_controller_lb_ip_address" { + value = length(var.ingress_whitelist) > 0 ? google_compute_address.worker_lb.0.address : "" +} + +output "control_plane_lb_ip_address" { + value = length(var.api_server_whitelist) > 0 ? google_compute_forwarding_rule.master_lb.0.ip_address : "" +} diff --git a/kubespray/contrib/terraform/gcp/modules/kubernetes-cluster/variables.tf b/kubespray/contrib/terraform/gcp/modules/kubernetes-cluster/variables.tf new file mode 100644 index 0000000..bb8d23b --- /dev/null +++ b/kubespray/contrib/terraform/gcp/modules/kubernetes-cluster/variables.tf @@ -0,0 +1,86 @@ +variable "region" { + type = string +} + +variable "prefix" {} + +variable "machines" { + type = map(object({ + node_type = string + size = string + zone = string + additional_disks = map(object({ + size = number + })) + boot_disk = object({ + image_name = string + size = number + }) + })) +} + +variable "master_sa_email" { + type = string +} + +variable "master_sa_scopes" { + type = list(string) +} + +variable "master_preemptible" { + type = bool +} + +variable "master_additional_disk_type" { + type = string +} + +variable "worker_sa_email" { + type = string +} + +variable "worker_sa_scopes" { + type = list(string) +} + +variable "worker_preemptible" { + type = bool +} + +variable "worker_additional_disk_type" { + type = string +} + +variable "ssh_pub_key" {} + +variable "ssh_whitelist" { + type = list(string) +} + +variable "api_server_whitelist" { + type = list(string) +} + +variable "nodeport_whitelist" { + type = list(string) +} + +variable "ingress_whitelist" { + type = list(string) + default = ["0.0.0.0/0"] +} + +variable "private_network_cidr" { + default = "10.0.10.0/24" +} + +variable "extra_ingress_firewalls" { + type = map(object({ + source_ranges = set(string) + protocol = string + ports = list(string) + target_tags = set(string) + })) + + default = {} +} diff --git a/kubespray/contrib/terraform/gcp/output.tf b/kubespray/contrib/terraform/gcp/output.tf new file mode 100644 index 0000000..09bf7fa --- /dev/null +++ b/kubespray/contrib/terraform/gcp/output.tf @@ -0,0 +1,15 @@ +output "master_ips" { + value = module.kubernetes.master_ip_addresses +} + +output "worker_ips" { + value = module.kubernetes.worker_ip_addresses +} + +output "ingress_controller_lb_ip_address" { + value = module.kubernetes.ingress_controller_lb_ip_address +} + +output "control_plane_lb_ip_address" { + value = module.kubernetes.control_plane_lb_ip_address +} diff --git a/kubespray/contrib/terraform/gcp/tfvars.json b/kubespray/contrib/terraform/gcp/tfvars.json new file mode 100644 index 0000000..056b8fe --- /dev/null +++ b/kubespray/contrib/terraform/gcp/tfvars.json @@ -0,0 +1,63 @@ +{ + "gcp_project_id": "GCP_PROJECT_ID", + "region": "us-central1", + "ssh_pub_key": "~/.ssh/id_rsa.pub", + + "keyfile_location": "service-account.json", + + "prefix": "development", + + "ssh_whitelist": [ + "1.2.3.4/32" + ], + "api_server_whitelist": [ + "1.2.3.4/32" + ], + "nodeport_whitelist": [ + "1.2.3.4/32" + ], + "ingress_whitelist": [ + "0.0.0.0/0" + ], + + "machines": { + "master-0": { + "node_type": "master", + "size": "n1-standard-2", + "zone": "us-central1-a", + "additional_disks": {}, + "boot_disk": { + "image_name": "ubuntu-os-cloud/ubuntu-2004-focal-v20220118", + "size": 50 + } + }, + "worker-0": { + "node_type": "worker", + "size": "n1-standard-8", + "zone": "us-central1-a", + "additional_disks": { + "extra-disk-1": { + "size": 100 + } + }, + "boot_disk": { + "image_name": "ubuntu-os-cloud/ubuntu-2004-focal-v20220118", + "size": 50 + } + }, + "worker-1": { + "node_type": "worker", + "size": "n1-standard-8", + "zone": "us-central1-a", + "additional_disks": { + "extra-disk-1": { + "size": 100 + } + }, + "boot_disk": { + "image_name": "ubuntu-os-cloud/ubuntu-2004-focal-v20220118", + "size": 50 + } + } + } +} diff --git a/kubespray/contrib/terraform/gcp/variables.tf b/kubespray/contrib/terraform/gcp/variables.tf new file mode 100644 index 0000000..3e96023 --- /dev/null +++ b/kubespray/contrib/terraform/gcp/variables.tf @@ -0,0 +1,108 @@ +variable keyfile_location { + description = "Location of the json keyfile to use with the google provider" + type = string +} + +variable region { + description = "Region of all resources" + type = string +} + +variable gcp_project_id { + description = "ID of the project" + type = string +} + +variable prefix { + description = "Prefix for resource names" + default = "default" +} + +variable machines { + description = "Cluster machines" + type = map(object({ + node_type = string + size = string + zone = string + additional_disks = map(object({ + size = number + })) + boot_disk = object({ + image_name = string + size = number + }) + })) +} + +variable "master_sa_email" { + type = string + default = "" +} + +variable "master_sa_scopes" { + type = list(string) + default = ["https://www.googleapis.com/auth/cloud-platform"] +} + +variable "master_preemptible" { + type = bool + default = false +} + +variable "master_additional_disk_type" { + type = string + default = "pd-ssd" +} + +variable "worker_sa_email" { + type = string + default = "" +} + +variable "worker_sa_scopes" { + type = list(string) + default = ["https://www.googleapis.com/auth/cloud-platform"] +} + +variable "worker_preemptible" { + type = bool + default = false +} + +variable "worker_additional_disk_type" { + type = string + default = "pd-ssd" +} + +variable ssh_pub_key { + description = "Path to public SSH key file which is injected into the VMs." + type = string +} + +variable ssh_whitelist { + type = list(string) +} + +variable api_server_whitelist { + type = list(string) +} + +variable nodeport_whitelist { + type = list(string) +} + +variable "ingress_whitelist" { + type = list(string) + default = ["0.0.0.0/0"] +} + +variable "extra_ingress_firewalls" { + type = map(object({ + source_ranges = set(string) + protocol = string + ports = list(string) + target_tags = set(string) + })) + + default = {} +} diff --git a/kubespray/contrib/terraform/group_vars b/kubespray/contrib/terraform/group_vars new file mode 120000 index 0000000..4dd828e --- /dev/null +++ b/kubespray/contrib/terraform/group_vars @@ -0,0 +1 @@ +../../inventory/local/group_vars \ No newline at end of file diff --git a/kubespray/contrib/terraform/hetzner/README.md b/kubespray/contrib/terraform/hetzner/README.md new file mode 100644 index 0000000..63ef640 --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/README.md @@ -0,0 +1,121 @@ +# Kubernetes on Hetzner with Terraform + +Provision a Kubernetes cluster on [Hetzner](https://www.hetzner.com/cloud) using Terraform and Kubespray + +## Overview + +The setup looks like following + +```text + Kubernetes cluster ++--------------------------+ +| +--------------+ | +| | +--------------+ | +| --> | | | | +| | | Master/etcd | | +| | | node(s) | | +| +-+ | | +| +--------------+ | +| ^ | +| | | +| v | +| +--------------+ | +| | +--------------+ | +| --> | | | | +| | | Worker | | +| | | node(s) | | +| +-+ | | +| +--------------+ | ++--------------------------+ +``` + +The nodes uses a private network for node to node communication and a public interface for all external communication. + +## Requirements + +* Terraform 0.14.0 or newer + +## Quickstart + +NOTE: Assumes you are at the root of the kubespray repo. + +For authentication in your cluster you can use the environment variables. + +```bash +export HCLOUD_TOKEN=api-token +``` + +Copy the cluster configuration file. + +```bash +CLUSTER=my-hetzner-cluster +cp -r inventory/sample inventory/$CLUSTER +cp contrib/terraform/hetzner/default.tfvars inventory/$CLUSTER/ +cd inventory/$CLUSTER +``` + +Edit `default.tfvars` to match your requirement. + +Flatcar Container Linux instead of the basic Hetzner Images. + +```bash +cd ../../contrib/terraform/hetzner +``` + +Edit `main.tf` and reactivate the module `source = "./modules/kubernetes-cluster-flatcar"`and +comment out the `#source = "./modules/kubernetes-cluster"`. + +activate `ssh_private_key_path = var.ssh_private_key_path`. The VM boots into +Rescue-Mode with the selected image of the `var.machines` but installs Flatcar instead. + +Run Terraform to create the infrastructure. + +```bash +cd ./kubespray +terraform -chdir=./contrib/terraform/hetzner/ init +terraform -chdir=./contrib/terraform/hetzner/ apply --var-file=../../../inventory/$CLUSTER/default.tfvars +``` + +You should now have a inventory file named `inventory.ini` that you can use with kubespray. +You can use the inventory file with kubespray to set up a cluster. + +It is a good idea to check that you have basic SSH connectivity to the nodes. You can do that by: + +```bash +ansible -i inventory.ini -m ping all +``` + +You can setup Kubernetes with kubespray using the generated inventory: + +```bash +ansible-playbook -i inventory.ini ../../cluster.yml -b -v +``` + +## Cloud controller + +For better support with the cloud you can install the [hcloud cloud controller](https://github.com/hetznercloud/hcloud-cloud-controller-manager) and [CSI driver](https://github.com/hetznercloud/csi-driver). + +Please read the instructions in both repos on how to install it. + +## Teardown + +You can teardown your infrastructure using the following Terraform command: + +```bash +terraform destroy --var-file default.tfvars ../../contrib/terraform/hetzner +``` + +## Variables + +* `prefix`: Prefix to add to all resources, if set to "" don't set any prefix +* `ssh_public_keys`: List of public SSH keys to install on all machines +* `zone`: The zone where to run the cluster +* `network_zone`: the network zone where the cluster is running +* `machines`: Machines to provision. Key of this object will be used as the name of the machine + * `node_type`: The role of this node *(master|worker)* + * `size`: Size of the VM + * `image`: The image to use for the VM +* `ssh_whitelist`: List of IP ranges (CIDR) that will be allowed to ssh to the nodes +* `api_server_whitelist`: List of IP ranges (CIDR) that will be allowed to connect to the API server +* `nodeport_whitelist`: List of IP ranges (CIDR) that will be allowed to connect to the kubernetes nodes on port 30000-32767 (kubernetes nodeports) +* `ingress_whitelist`: List of IP ranges (CIDR) that will be allowed to connect to kubernetes workers on port 80 and 443 diff --git a/kubespray/contrib/terraform/hetzner/default.tfvars b/kubespray/contrib/terraform/hetzner/default.tfvars new file mode 100644 index 0000000..4e70bf1 --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/default.tfvars @@ -0,0 +1,46 @@ +prefix = "default" +zone = "hel1" +network_zone = "eu-central" +inventory_file = "inventory.ini" + +ssh_public_keys = [ + # Put your public SSH key here + "ssh-rsa I-did-not-read-the-docs", + "ssh-rsa I-did-not-read-the-docs 2", +] + +ssh_private_key_path = "~/.ssh/id_rsa" + +machines = { + "master-0" : { + "node_type" : "master", + "size" : "cx21", + "image" : "ubuntu-22.04", + }, + "worker-0" : { + "node_type" : "worker", + "size" : "cx21", + "image" : "ubuntu-22.04", + }, + "worker-1" : { + "node_type" : "worker", + "size" : "cx21", + "image" : "ubuntu-22.04", + } +} + +nodeport_whitelist = [ + "0.0.0.0/0" +] + +ingress_whitelist = [ + "0.0.0.0/0" +] + +ssh_whitelist = [ + "0.0.0.0/0" +] + +api_server_whitelist = [ + "0.0.0.0/0" +] diff --git a/kubespray/contrib/terraform/hetzner/main.tf b/kubespray/contrib/terraform/hetzner/main.tf new file mode 100644 index 0000000..8e38cee --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/main.tf @@ -0,0 +1,57 @@ +provider "hcloud" {} + +module "kubernetes" { + source = "./modules/kubernetes-cluster" + # source = "./modules/kubernetes-cluster-flatcar" + + prefix = var.prefix + + zone = var.zone + + machines = var.machines + + #only for flatcar + #ssh_private_key_path = var.ssh_private_key_path + + ssh_public_keys = var.ssh_public_keys + network_zone = var.network_zone + + ssh_whitelist = var.ssh_whitelist + api_server_whitelist = var.api_server_whitelist + nodeport_whitelist = var.nodeport_whitelist + ingress_whitelist = var.ingress_whitelist +} + +# +# Generate ansible inventory +# + +locals { + inventory = templatefile( + "${path.module}/templates/inventory.tpl", + { + connection_strings_master = join("\n", formatlist("%s ansible_user=ubuntu ansible_host=%s ip=%s etcd_member_name=etcd%d", + keys(module.kubernetes.master_ip_addresses), + values(module.kubernetes.master_ip_addresses).*.public_ip, + values(module.kubernetes.master_ip_addresses).*.private_ip, + range(1, length(module.kubernetes.master_ip_addresses) + 1))) + connection_strings_worker = join("\n", formatlist("%s ansible_user=ubuntu ansible_host=%s ip=%s", + keys(module.kubernetes.worker_ip_addresses), + values(module.kubernetes.worker_ip_addresses).*.public_ip, + values(module.kubernetes.worker_ip_addresses).*.private_ip)) + list_master = join("\n", keys(module.kubernetes.master_ip_addresses)) + list_worker = join("\n", keys(module.kubernetes.worker_ip_addresses)) + network_id = module.kubernetes.network_id + } + ) +} + +resource "null_resource" "inventories" { + provisioner "local-exec" { + command = "echo '${local.inventory}' > ${var.inventory_file}" + } + + triggers = { + template = local.inventory + } +} diff --git a/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/main.tf b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/main.tf new file mode 100644 index 0000000..b54d360 --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/main.tf @@ -0,0 +1,144 @@ +resource "hcloud_network" "kubernetes" { + name = "${var.prefix}-network" + ip_range = var.private_network_cidr +} + +resource "hcloud_network_subnet" "kubernetes" { + type = "cloud" + network_id = hcloud_network.kubernetes.id + network_zone = var.network_zone + ip_range = var.private_subnet_cidr +} + +resource "hcloud_ssh_key" "first" { + name = var.prefix + public_key = var.ssh_public_keys.0 +} + +resource "hcloud_server" "machine" { + for_each = { + for name, machine in var.machines : + name => machine + } + + name = "${var.prefix}-${each.key}" + ssh_keys = [hcloud_ssh_key.first.id] + # boot into rescue OS + rescue = "linux64" + # dummy value for the OS because Flatcar is not available + image = each.value.image + server_type = each.value.size + location = var.zone + connection { + host = self.ipv4_address + timeout = "5m" + private_key = file(var.ssh_private_key_path) + } + firewall_ids = each.value.node_type == "master" ? [hcloud_firewall.master.id] : [hcloud_firewall.worker.id] + provisioner "file" { + content = data.ct_config.machine-ignitions[each.key].rendered + destination = "/root/ignition.json" + } + + provisioner "remote-exec" { + inline = [ + "set -ex", + "apt update", + "apt install -y gawk", + "curl -fsSLO --retry-delay 1 --retry 60 --retry-connrefused --retry-max-time 60 --connect-timeout 20 https://raw.githubusercontent.com/flatcar/init/flatcar-master/bin/flatcar-install", + "chmod +x flatcar-install", + "./flatcar-install -s -i /root/ignition.json -C stable", + "shutdown -r +1", + ] + } + + # optional: + provisioner "remote-exec" { + connection { + host = self.ipv4_address + private_key = file(var.ssh_private_key_path) + timeout = "3m" + user = var.user_flatcar + } + + inline = [ + "sudo hostnamectl set-hostname ${self.name}", + ] + } +} + +resource "hcloud_server_network" "machine" { + for_each = { + for name, machine in var.machines : + name => hcloud_server.machine[name] + } + server_id = each.value.id + subnet_id = hcloud_network_subnet.kubernetes.id +} + +data "ct_config" "machine-ignitions" { + for_each = { + for name, machine in var.machines : + name => machine + } + + strict = false + content = templatefile( + "${path.module}/templates/machine.yaml.tmpl", + { + ssh_keys = jsonencode(var.ssh_public_keys) + user_flatcar = var.user_flatcar + name = each.key + } + ) +} + +resource "hcloud_firewall" "master" { + name = "${var.prefix}-master-firewall" + + rule { + direction = "in" + protocol = "tcp" + port = "22" + source_ips = var.ssh_whitelist + } + + rule { + direction = "in" + protocol = "tcp" + port = "6443" + source_ips = var.api_server_whitelist + } +} + +resource "hcloud_firewall" "worker" { + name = "${var.prefix}-worker-firewall" + + rule { + direction = "in" + protocol = "tcp" + port = "22" + source_ips = var.ssh_whitelist + } + + rule { + direction = "in" + protocol = "tcp" + port = "80" + source_ips = var.ingress_whitelist + } + + rule { + direction = "in" + protocol = "tcp" + port = "443" + source_ips = var.ingress_whitelist + } + + rule { + direction = "in" + protocol = "tcp" + port = "30000-32767" + source_ips = var.nodeport_whitelist + } +} diff --git a/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/outputs.tf b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/outputs.tf new file mode 100644 index 0000000..be524de --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/outputs.tf @@ -0,0 +1,29 @@ +output "master_ip_addresses" { + value = { + for name, machine in var.machines : + name => { + "private_ip" = hcloud_server_network.machine[name].ip + "public_ip" = hcloud_server.machine[name].ipv4_address + } + if machine.node_type == "master" + } +} + +output "worker_ip_addresses" { + value = { + for name, machine in var.machines : + name => { + "private_ip" = hcloud_server_network.machine[name].ip + "public_ip" = hcloud_server.machine[name].ipv4_address + } + if machine.node_type == "worker" + } +} + +output "cluster_private_network_cidr" { + value = var.private_subnet_cidr +} + +output "network_id" { + value = hcloud_network.kubernetes.id +} diff --git a/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/templates/machine.yaml.tmpl b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/templates/machine.yaml.tmpl new file mode 100644 index 0000000..95ce1d8 --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/templates/machine.yaml.tmpl @@ -0,0 +1,19 @@ +variant: flatcar +version: 1.0.0 + +passwd: + users: + - name: ${user_flatcar} + ssh_authorized_keys: ${ssh_keys} + +storage: + files: + - path: /home/core/works + filesystem: root + mode: 0755 + contents: + inline: | + #!/bin/bash + set -euo pipefail + hostname="$(hostname)" + echo My name is ${name} and the hostname is $${hostname} diff --git a/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/variables.tf b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/variables.tf new file mode 100644 index 0000000..8093779 --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/variables.tf @@ -0,0 +1,60 @@ + +variable "zone" { + type = string + default = "fsn1" +} + +variable "prefix" { + default = "k8s" +} + +variable "user_flatcar" { + type = string + default = "core" +} + +variable "machines" { + type = map(object({ + node_type = string + size = string + image = string + })) +} + + + +variable "ssh_public_keys" { + type = list(string) +} + +variable "ssh_private_key_path" { + type = string + default = "~/.ssh/id_rsa" +} + +variable "ssh_whitelist" { + type = list(string) +} + +variable "api_server_whitelist" { + type = list(string) +} + +variable "nodeport_whitelist" { + type = list(string) +} + +variable "ingress_whitelist" { + type = list(string) +} + +variable "private_network_cidr" { + default = "10.0.0.0/16" +} + +variable "private_subnet_cidr" { + default = "10.0.10.0/24" +} +variable "network_zone" { + default = "eu-central" +} diff --git a/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/versions.tf b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/versions.tf new file mode 100644 index 0000000..ac98e27 --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster-flatcar/versions.tf @@ -0,0 +1,14 @@ +terraform { + required_providers { + hcloud = { + source = "hetznercloud/hcloud" + } + ct = { + source = "poseidon/ct" + version = "0.11.0" + } + null = { + source = "hashicorp/null" + } + } +} diff --git a/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/main.tf b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/main.tf new file mode 100644 index 0000000..2a0e458 --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/main.tf @@ -0,0 +1,122 @@ +resource "hcloud_network" "kubernetes" { + name = "${var.prefix}-network" + ip_range = var.private_network_cidr +} + +resource "hcloud_network_subnet" "kubernetes" { + type = "cloud" + network_id = hcloud_network.kubernetes.id + network_zone = var.network_zone + ip_range = var.private_subnet_cidr +} + +resource "hcloud_server" "master" { + for_each = { + for name, machine in var.machines : + name => machine + if machine.node_type == "master" + } + + name = "${var.prefix}-${each.key}" + image = each.value.image + server_type = each.value.size + location = var.zone + + user_data = templatefile( + "${path.module}/templates/cloud-init.tmpl", + { + ssh_public_keys = var.ssh_public_keys + } + ) + + firewall_ids = [hcloud_firewall.master.id] +} + +resource "hcloud_server_network" "master" { + for_each = hcloud_server.master + + server_id = each.value.id + + subnet_id = hcloud_network_subnet.kubernetes.id +} + +resource "hcloud_server" "worker" { + for_each = { + for name, machine in var.machines : + name => machine + if machine.node_type == "worker" + } + + name = "${var.prefix}-${each.key}" + image = each.value.image + server_type = each.value.size + location = var.zone + + user_data = templatefile( + "${path.module}/templates/cloud-init.tmpl", + { + ssh_public_keys = var.ssh_public_keys + } + ) + + firewall_ids = [hcloud_firewall.worker.id] + +} + +resource "hcloud_server_network" "worker" { + for_each = hcloud_server.worker + + server_id = each.value.id + + subnet_id = hcloud_network_subnet.kubernetes.id +} + +resource "hcloud_firewall" "master" { + name = "${var.prefix}-master-firewall" + + rule { + direction = "in" + protocol = "tcp" + port = "22" + source_ips = var.ssh_whitelist + } + + rule { + direction = "in" + protocol = "tcp" + port = "6443" + source_ips = var.api_server_whitelist + } +} + +resource "hcloud_firewall" "worker" { + name = "${var.prefix}-worker-firewall" + + rule { + direction = "in" + protocol = "tcp" + port = "22" + source_ips = var.ssh_whitelist + } + + rule { + direction = "in" + protocol = "tcp" + port = "80" + source_ips = var.ingress_whitelist + } + + rule { + direction = "in" + protocol = "tcp" + port = "443" + source_ips = var.ingress_whitelist + } + + rule { + direction = "in" + protocol = "tcp" + port = "30000-32767" + source_ips = var.nodeport_whitelist + } +} diff --git a/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/output.tf b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/output.tf new file mode 100644 index 0000000..5c31aaa --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/output.tf @@ -0,0 +1,27 @@ +output "master_ip_addresses" { + value = { + for key, instance in hcloud_server.master : + instance.name => { + "private_ip" = hcloud_server_network.master[key].ip + "public_ip" = hcloud_server.master[key].ipv4_address + } + } +} + +output "worker_ip_addresses" { + value = { + for key, instance in hcloud_server.worker : + instance.name => { + "private_ip" = hcloud_server_network.worker[key].ip + "public_ip" = hcloud_server.worker[key].ipv4_address + } + } +} + +output "cluster_private_network_cidr" { + value = var.private_subnet_cidr +} + +output "network_id" { + value = hcloud_network.kubernetes.id +} diff --git a/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/templates/cloud-init.tmpl b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/templates/cloud-init.tmpl new file mode 100644 index 0000000..02a4e2d --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/templates/cloud-init.tmpl @@ -0,0 +1,16 @@ +#cloud-config + +users: + - default + - name: ubuntu + shell: /bin/bash + sudo: "ALL=(ALL) NOPASSWD:ALL" + ssh_authorized_keys: + %{ for ssh_public_key in ssh_public_keys ~} + - ${ssh_public_key} + %{ endfor ~} + +ssh_authorized_keys: +%{ for ssh_public_key in ssh_public_keys ~} + - ${ssh_public_key} +%{ endfor ~} diff --git a/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/variables.tf b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/variables.tf new file mode 100644 index 0000000..7486e08 --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/variables.tf @@ -0,0 +1,44 @@ +variable "zone" { + type = string +} + +variable "prefix" {} + +variable "machines" { + type = map(object({ + node_type = string + size = string + image = string + })) +} + +variable "ssh_public_keys" { + type = list(string) +} + +variable "ssh_whitelist" { + type = list(string) +} + +variable "api_server_whitelist" { + type = list(string) +} + +variable "nodeport_whitelist" { + type = list(string) +} + +variable "ingress_whitelist" { + type = list(string) +} + +variable "private_network_cidr" { + default = "10.0.0.0/16" +} + +variable "private_subnet_cidr" { + default = "10.0.10.0/24" +} +variable "network_zone" { + default = "eu-central" +} diff --git a/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/versions.tf b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/versions.tf new file mode 100644 index 0000000..78bc504 --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/modules/kubernetes-cluster/versions.tf @@ -0,0 +1,9 @@ +terraform { + required_providers { + hcloud = { + source = "hetznercloud/hcloud" + version = "1.38.2" + } + } + required_version = ">= 0.14" +} diff --git a/kubespray/contrib/terraform/hetzner/output.tf b/kubespray/contrib/terraform/hetzner/output.tf new file mode 100644 index 0000000..0336f72 --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/output.tf @@ -0,0 +1,7 @@ +output "master_ips" { + value = module.kubernetes.master_ip_addresses +} + +output "worker_ips" { + value = module.kubernetes.worker_ip_addresses +} diff --git a/kubespray/contrib/terraform/hetzner/sample-inventory/cluster.tfvars b/kubespray/contrib/terraform/hetzner/sample-inventory/cluster.tfvars new file mode 100644 index 0000000..4e70bf1 --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/sample-inventory/cluster.tfvars @@ -0,0 +1,46 @@ +prefix = "default" +zone = "hel1" +network_zone = "eu-central" +inventory_file = "inventory.ini" + +ssh_public_keys = [ + # Put your public SSH key here + "ssh-rsa I-did-not-read-the-docs", + "ssh-rsa I-did-not-read-the-docs 2", +] + +ssh_private_key_path = "~/.ssh/id_rsa" + +machines = { + "master-0" : { + "node_type" : "master", + "size" : "cx21", + "image" : "ubuntu-22.04", + }, + "worker-0" : { + "node_type" : "worker", + "size" : "cx21", + "image" : "ubuntu-22.04", + }, + "worker-1" : { + "node_type" : "worker", + "size" : "cx21", + "image" : "ubuntu-22.04", + } +} + +nodeport_whitelist = [ + "0.0.0.0/0" +] + +ingress_whitelist = [ + "0.0.0.0/0" +] + +ssh_whitelist = [ + "0.0.0.0/0" +] + +api_server_whitelist = [ + "0.0.0.0/0" +] diff --git a/kubespray/contrib/terraform/hetzner/sample-inventory/group_vars b/kubespray/contrib/terraform/hetzner/sample-inventory/group_vars new file mode 120000 index 0000000..3735958 --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/sample-inventory/group_vars @@ -0,0 +1 @@ +../../../../inventory/sample/group_vars \ No newline at end of file diff --git a/kubespray/contrib/terraform/hetzner/templates/inventory.tpl b/kubespray/contrib/terraform/hetzner/templates/inventory.tpl new file mode 100644 index 0000000..56666e1 --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/templates/inventory.tpl @@ -0,0 +1,19 @@ +[all] +${connection_strings_master} +${connection_strings_worker} + +[kube_control_plane] +${list_master} + +[etcd] +${list_master} + +[kube_node] +${list_worker} + +[k8s_cluster:children] +kube-master +kube-node + +[k8s_cluster:vars] +network_id=${network_id} diff --git a/kubespray/contrib/terraform/hetzner/variables.tf b/kubespray/contrib/terraform/hetzner/variables.tf new file mode 100644 index 0000000..049ce0d --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/variables.tf @@ -0,0 +1,56 @@ +variable "zone" { + description = "The zone where to run the cluster" +} +variable "network_zone" { + description = "The network zone where the cluster is running" + default = "eu-central" +} + +variable "prefix" { + description = "Prefix for resource names" + default = "default" +} + +variable "machines" { + description = "Cluster machines" + type = map(object({ + node_type = string + size = string + image = string + })) +} + +variable "ssh_public_keys" { + description = "Public SSH key which are injected into the VMs." + type = list(string) +} + +variable "ssh_private_key_path" { + description = "Private SSH key which connect to the VMs." + type = string + default = "~/.ssh/id_rsa" +} + +variable "ssh_whitelist" { + description = "List of IP ranges (CIDR) to whitelist for ssh" + type = list(string) +} + +variable "api_server_whitelist" { + description = "List of IP ranges (CIDR) to whitelist for kubernetes api server" + type = list(string) +} + +variable "nodeport_whitelist" { + description = "List of IP ranges (CIDR) to whitelist for kubernetes nodeports" + type = list(string) +} + +variable "ingress_whitelist" { + description = "List of IP ranges (CIDR) to whitelist for HTTP" + type = list(string) +} + +variable "inventory_file" { + description = "Where to store the generated inventory file" +} diff --git a/kubespray/contrib/terraform/hetzner/versions.tf b/kubespray/contrib/terraform/hetzner/versions.tf new file mode 100644 index 0000000..e331beb --- /dev/null +++ b/kubespray/contrib/terraform/hetzner/versions.tf @@ -0,0 +1,12 @@ +terraform { + required_providers { + hcloud = { + source = "hetznercloud/hcloud" + version = "1.38.2" + } + null = { + source = "hashicorp/null" + } + } + required_version = ">= 0.14" +} diff --git a/kubespray/contrib/terraform/nifcloud/.gitignore b/kubespray/contrib/terraform/nifcloud/.gitignore new file mode 100644 index 0000000..9adadc3 --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/.gitignore @@ -0,0 +1,5 @@ +*.tfstate* +.terraform.lock.hcl +.terraform + +sample-inventory/inventory.ini diff --git a/kubespray/contrib/terraform/nifcloud/README.md b/kubespray/contrib/terraform/nifcloud/README.md new file mode 100644 index 0000000..8c46df4 --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/README.md @@ -0,0 +1,137 @@ +# Kubernetes on NIFCLOUD with Terraform + +Provision a Kubernetes cluster on [NIFCLOUD](https://pfs.nifcloud.com/) using Terraform and Kubespray + +## Overview + +The setup looks like following + +```text + Kubernetes cluster + +----------------------------+ ++---------------+ | +--------------------+ | +| | | | +--------------------+ | +| API server LB +---------> | | | | +| | | | | Control Plane/etcd | | ++---------------+ | | | node(s) | | + | +-+ | | + | +--------------------+ | + | ^ | + | | | + | v | + | +--------------------+ | + | | +--------------------+ | + | | | | | + | | | Worker | | + | | | node(s) | | + | +-+ | | + | +--------------------+ | + +----------------------------+ +``` + +## Requirements + +* Terraform 1.3.7 + +## Quickstart + +### Export Variables + +* Your NIFCLOUD credentials: + + ```bash + export NIFCLOUD_ACCESS_KEY_ID= + export NIFCLOUD_SECRET_ACCESS_KEY= + ``` + +* The SSH KEY used to connect to the instance: + * FYI: [Cloud Help(SSH Key)](https://pfs.nifcloud.com/help/ssh.htm) + + ```bash + export TF_VAR_SSHKEY_NAME= + ``` + +* The IP address to connect to bastion server: + + ```bash + export TF_VAR_working_instance_ip=$(curl ifconfig.me) + ``` + +### Create The Infrastructure + +* Run terraform: + + ```bash + terraform init + terraform apply -var-file ./sample-inventory/cluster.tfvars + ``` + +### Setup The Kubernetes + +* Generate cluster configuration file: + + ```bash + ./generate-inventory.sh > sample-inventory/inventory.ini + +* Export Variables: + + ```bash + BASTION_IP=$(terraform output -json | jq -r '.kubernetes_cluster.value.bastion_info | to_entries[].value.public_ip') + API_LB_IP=$(terraform output -json | jq -r '.kubernetes_cluster.value.control_plane_lb') + CP01_IP=$(terraform output -json | jq -r '.kubernetes_cluster.value.control_plane_info | to_entries[0].value.private_ip') + export ANSIBLE_SSH_ARGS="-o StrictHostKeyChecking=no -o UserKnownHostsFile=/dev/null -o ProxyCommand=\"ssh root@${BASTION_IP} -W %h:%p\"" + ``` + +* Set ssh-agent" + + ```bash + eval `ssh-agent` + ssh-add + ``` + +* Run cluster.yml playbook: + + ```bash + cd ./../../../ + ansible-playbook -i contrib/terraform/nifcloud/inventory/inventory.ini cluster.yml + ``` + +### Connecting to Kubernetes + +* [Install kubectl](https://kubernetes.io/docs/tasks/tools/) on the localhost +* Fetching kubeconfig file: + + ```bash + mkdir -p ~/.kube + scp -o ProxyCommand="ssh root@${BASTION_IP} -W %h:%p" root@${CP01_IP}:/etc/kubernetes/admin.conf ~/.kube/config + ``` + +* Rewrite /etc/hosts + + ```bash + sudo echo "${API_LB_IP} lb-apiserver.kubernetes.local" >> /etc/hosts + ``` + +* Run kubectl + + ```bash + kubectl get node + ``` + +## Variables + +* `region`: Region where to run the cluster +* `az`: Availability zone where to run the cluster +* `private_ip_bn`: Private ip address of bastion server +* `private_network_cidr`: Subnet of private network +* `instances_cp`: Machine to provision as Control Plane. Key of this object will be used as part of the machine' name + * `private_ip`: private ip address of machine +* `instances_wk`: Machine to provision as Worker Node. Key of this object will be used as part of the machine' name + * `private_ip`: private ip address of machine +* `instance_key_name`: The key name of the Key Pair to use for the instance +* `instance_type_bn`: The instance type of bastion server +* `instance_type_wk`: The instance type of worker node +* `instance_type_cp`: The instance type of control plane +* `image_name`: OS image used for the instance +* `working_instance_ip`: The IP address to connect to bastion server +* `accounting_type`: Accounting type. (1: monthly, 2: pay per use) diff --git a/kubespray/contrib/terraform/nifcloud/generate-inventory.sh b/kubespray/contrib/terraform/nifcloud/generate-inventory.sh new file mode 100755 index 0000000..5d90eb5 --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/generate-inventory.sh @@ -0,0 +1,64 @@ +#!/bin/bash + +# +# Generates a inventory file based on the terraform output. +# After provisioning a cluster, simply run this command and supply the terraform state file +# Default state file is terraform.tfstate +# + +set -e + +TF_OUT=$(terraform output -json) + +CONTROL_PLANES=$(jq -r '.kubernetes_cluster.value.control_plane_info | to_entries[]' <(echo "${TF_OUT}")) +WORKERS=$(jq -r '.kubernetes_cluster.value.worker_info | to_entries[]' <(echo "${TF_OUT}")) +mapfile -t CONTROL_PLANE_NAMES < <(jq -r '.key' <(echo "${CONTROL_PLANES}")) +mapfile -t WORKER_NAMES < <(jq -r '.key' <(echo "${WORKERS}")) + +API_LB=$(jq -r '.kubernetes_cluster.value.control_plane_lb' <(echo "${TF_OUT}")) + +echo "[all]" +# Generate control plane hosts +i=1 +for name in "${CONTROL_PLANE_NAMES[@]}"; do + private_ip=$(jq -r '. | select( .key=='"\"${name}\""' ) | .value.private_ip' <(echo "${CONTROL_PLANES}")) + echo "${name} ansible_user=root ansible_host=${private_ip} access_ip=${private_ip} ip=${private_ip} etcd_member_name=etcd${i}" + i=$(( i + 1 )) +done + +# Generate worker hosts +for name in "${WORKER_NAMES[@]}"; do + private_ip=$(jq -r '. | select( .key=='"\"${name}\""' ) | .value.private_ip' <(echo "${WORKERS}")) + echo "${name} ansible_user=root ansible_host=${private_ip} access_ip=${private_ip} ip=${private_ip}" +done + +API_LB=$(jq -r '.kubernetes_cluster.value.control_plane_lb' <(echo "${TF_OUT}")) + +echo "" +echo "[all:vars]" +echo "upstream_dns_servers=['8.8.8.8','8.8.4.4']" +echo "loadbalancer_apiserver={'address':'${API_LB}','port':'6443'}" + + +echo "" +echo "[kube_control_plane]" +for name in "${CONTROL_PLANE_NAMES[@]}"; do + echo "${name}" +done + +echo "" +echo "[etcd]" +for name in "${CONTROL_PLANE_NAMES[@]}"; do + echo "${name}" +done + +echo "" +echo "[kube_node]" +for name in "${WORKER_NAMES[@]}"; do + echo "${name}" +done + +echo "" +echo "[k8s_cluster:children]" +echo "kube_control_plane" +echo "kube_node" diff --git a/kubespray/contrib/terraform/nifcloud/main.tf b/kubespray/contrib/terraform/nifcloud/main.tf new file mode 100644 index 0000000..d5a0709 --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/main.tf @@ -0,0 +1,36 @@ +provider "nifcloud" { + region = var.region +} + +module "kubernetes_cluster" { + source = "./modules/kubernetes-cluster" + + availability_zone = var.az + prefix = "dev" + + private_network_cidr = var.private_network_cidr + + instance_key_name = var.instance_key_name + instances_cp = var.instances_cp + instances_wk = var.instances_wk + image_name = var.image_name + + instance_type_bn = var.instance_type_bn + instance_type_cp = var.instance_type_cp + instance_type_wk = var.instance_type_wk + + private_ip_bn = var.private_ip_bn + + additional_lb_filter = [var.working_instance_ip] +} + +resource "nifcloud_security_group_rule" "ssh_from_bastion" { + security_group_names = [ + module.kubernetes_cluster.security_group_name.bastion + ] + type = "IN" + from_port = 22 + to_port = 22 + protocol = "TCP" + cidr_ip = var.working_instance_ip +} diff --git a/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/main.tf b/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/main.tf new file mode 100644 index 0000000..0e5fd38 --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/main.tf @@ -0,0 +1,301 @@ +################################################# +## +## Local variables +## +locals { + # e.g. east-11 is 11 + az_num = reverse(split("-", var.availability_zone))[0] + # e.g. east-11 is e11 + az_short_name = "${substr(reverse(split("-", var.availability_zone))[1], 0, 1)}${local.az_num}" + + # Port used by the protocol + port_ssh = 22 + port_kubectl = 6443 + port_kubelet = 10250 + + # calico: https://docs.tigera.io/calico/latest/getting-started/kubernetes/requirements#network-requirements + port_bgp = 179 + port_vxlan = 4789 + port_etcd = 2379 +} + +################################################# +## +## General +## + +# data +data "nifcloud_image" "this" { + image_name = var.image_name +} + +# private lan +resource "nifcloud_private_lan" "this" { + private_lan_name = "${var.prefix}lan" + availability_zone = var.availability_zone + cidr_block = var.private_network_cidr + accounting_type = var.accounting_type +} + +################################################# +## +## Bastion +## +resource "nifcloud_security_group" "bn" { + group_name = "${var.prefix}bn" + description = "${var.prefix} bastion" + availability_zone = var.availability_zone +} + +resource "nifcloud_instance" "bn" { + + instance_id = "${local.az_short_name}${var.prefix}bn01" + security_group = nifcloud_security_group.bn.group_name + instance_type = var.instance_type_bn + + user_data = templatefile("${path.module}/templates/userdata.tftpl", { + private_ip_address = var.private_ip_bn + ssh_port = local.port_ssh + hostname = "${local.az_short_name}${var.prefix}bn01" + }) + + availability_zone = var.availability_zone + accounting_type = var.accounting_type + image_id = data.nifcloud_image.this.image_id + key_name = var.instance_key_name + + network_interface { + network_id = "net-COMMON_GLOBAL" + } + network_interface { + network_id = nifcloud_private_lan.this.network_id + ip_address = "static" + } + + # The image_id changes when the OS image type is demoted from standard to public. + lifecycle { + ignore_changes = [ + image_id, + user_data, + ] + } +} + +################################################# +## +## Control Plane +## +resource "nifcloud_security_group" "cp" { + group_name = "${var.prefix}cp" + description = "${var.prefix} control plane" + availability_zone = var.availability_zone +} + +resource "nifcloud_instance" "cp" { + for_each = var.instances_cp + + instance_id = "${local.az_short_name}${var.prefix}${each.key}" + security_group = nifcloud_security_group.cp.group_name + instance_type = var.instance_type_cp + user_data = templatefile("${path.module}/templates/userdata.tftpl", { + private_ip_address = each.value.private_ip + ssh_port = local.port_ssh + hostname = "${local.az_short_name}${var.prefix}${each.key}" + }) + + availability_zone = var.availability_zone + accounting_type = var.accounting_type + image_id = data.nifcloud_image.this.image_id + key_name = var.instance_key_name + + network_interface { + network_id = "net-COMMON_GLOBAL" + } + network_interface { + network_id = nifcloud_private_lan.this.network_id + ip_address = "static" + } + + # The image_id changes when the OS image type is demoted from standard to public. + lifecycle { + ignore_changes = [ + image_id, + user_data, + ] + } +} + +resource "nifcloud_load_balancer" "this" { + load_balancer_name = "${local.az_short_name}${var.prefix}cp" + accounting_type = var.accounting_type + balancing_type = 1 // Round-Robin + load_balancer_port = local.port_kubectl + instance_port = local.port_kubectl + instances = [for v in nifcloud_instance.cp : v.instance_id] + filter = concat( + [for k, v in nifcloud_instance.cp : v.public_ip], + [for k, v in nifcloud_instance.wk : v.public_ip], + var.additional_lb_filter, + ) + filter_type = 1 // Allow +} + +################################################# +## +## Worker +## +resource "nifcloud_security_group" "wk" { + group_name = "${var.prefix}wk" + description = "${var.prefix} worker" + availability_zone = var.availability_zone +} + +resource "nifcloud_instance" "wk" { + for_each = var.instances_wk + + instance_id = "${local.az_short_name}${var.prefix}${each.key}" + security_group = nifcloud_security_group.wk.group_name + instance_type = var.instance_type_wk + user_data = templatefile("${path.module}/templates/userdata.tftpl", { + private_ip_address = each.value.private_ip + ssh_port = local.port_ssh + hostname = "${local.az_short_name}${var.prefix}${each.key}" + }) + + availability_zone = var.availability_zone + accounting_type = var.accounting_type + image_id = data.nifcloud_image.this.image_id + key_name = var.instance_key_name + + network_interface { + network_id = "net-COMMON_GLOBAL" + } + network_interface { + network_id = nifcloud_private_lan.this.network_id + ip_address = "static" + } + + # The image_id changes when the OS image type is demoted from standard to public. + lifecycle { + ignore_changes = [ + image_id, + user_data, + ] + } +} + +################################################# +## +## Security Group Rule: Kubernetes +## + +# ssh +resource "nifcloud_security_group_rule" "ssh_from_bastion" { + security_group_names = [ + nifcloud_security_group.wk.group_name, + nifcloud_security_group.cp.group_name, + ] + type = "IN" + from_port = local.port_ssh + to_port = local.port_ssh + protocol = "TCP" + source_security_group_name = nifcloud_security_group.bn.group_name +} + +# kubectl +resource "nifcloud_security_group_rule" "kubectl_from_worker" { + security_group_names = [ + nifcloud_security_group.cp.group_name, + ] + type = "IN" + from_port = local.port_kubectl + to_port = local.port_kubectl + protocol = "TCP" + source_security_group_name = nifcloud_security_group.wk.group_name +} + +# kubelet +resource "nifcloud_security_group_rule" "kubelet_from_worker" { + security_group_names = [ + nifcloud_security_group.cp.group_name, + ] + type = "IN" + from_port = local.port_kubelet + to_port = local.port_kubelet + protocol = "TCP" + source_security_group_name = nifcloud_security_group.wk.group_name +} + +resource "nifcloud_security_group_rule" "kubelet_from_control_plane" { + security_group_names = [ + nifcloud_security_group.wk.group_name, + ] + type = "IN" + from_port = local.port_kubelet + to_port = local.port_kubelet + protocol = "TCP" + source_security_group_name = nifcloud_security_group.cp.group_name +} + +################################################# +## +## Security Group Rule: calico +## + +# vslan +resource "nifcloud_security_group_rule" "vxlan_from_control_plane" { + security_group_names = [ + nifcloud_security_group.wk.group_name, + ] + type = "IN" + from_port = local.port_vxlan + to_port = local.port_vxlan + protocol = "UDP" + source_security_group_name = nifcloud_security_group.cp.group_name +} + +resource "nifcloud_security_group_rule" "vxlan_from_worker" { + security_group_names = [ + nifcloud_security_group.cp.group_name, + ] + type = "IN" + from_port = local.port_vxlan + to_port = local.port_vxlan + protocol = "UDP" + source_security_group_name = nifcloud_security_group.wk.group_name +} + +# bgp +resource "nifcloud_security_group_rule" "bgp_from_control_plane" { + security_group_names = [ + nifcloud_security_group.wk.group_name, + ] + type = "IN" + from_port = local.port_bgp + to_port = local.port_bgp + protocol = "TCP" + source_security_group_name = nifcloud_security_group.cp.group_name +} + +resource "nifcloud_security_group_rule" "bgp_from_worker" { + security_group_names = [ + nifcloud_security_group.cp.group_name, + ] + type = "IN" + from_port = local.port_bgp + to_port = local.port_bgp + protocol = "TCP" + source_security_group_name = nifcloud_security_group.wk.group_name +} + +# etcd +resource "nifcloud_security_group_rule" "etcd_from_worker" { + security_group_names = [ + nifcloud_security_group.cp.group_name, + ] + type = "IN" + from_port = local.port_etcd + to_port = local.port_etcd + protocol = "TCP" + source_security_group_name = nifcloud_security_group.wk.group_name +} diff --git a/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/outputs.tf b/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/outputs.tf new file mode 100644 index 0000000..a6232f8 --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/outputs.tf @@ -0,0 +1,48 @@ +output "control_plane_lb" { + description = "The DNS name of LB for control plane" + value = nifcloud_load_balancer.this.dns_name +} + +output "security_group_name" { + description = "The security group used in the cluster" + value = { + bastion = nifcloud_security_group.bn.group_name, + control_plane = nifcloud_security_group.cp.group_name, + worker = nifcloud_security_group.wk.group_name, + } +} + +output "private_network_id" { + description = "The private network used in the cluster" + value = nifcloud_private_lan.this.id +} + +output "bastion_info" { + description = "The basion information in cluster" + value = { (nifcloud_instance.bn.instance_id) : { + instance_id = nifcloud_instance.bn.instance_id, + unique_id = nifcloud_instance.bn.unique_id, + private_ip = nifcloud_instance.bn.private_ip, + public_ip = nifcloud_instance.bn.public_ip, + } } +} + +output "worker_info" { + description = "The worker information in cluster" + value = { for v in nifcloud_instance.wk : v.instance_id => { + instance_id = v.instance_id, + unique_id = v.unique_id, + private_ip = v.private_ip, + public_ip = v.public_ip, + } } +} + +output "control_plane_info" { + description = "The control plane information in cluster" + value = { for v in nifcloud_instance.cp : v.instance_id => { + instance_id = v.instance_id, + unique_id = v.unique_id, + private_ip = v.private_ip, + public_ip = v.public_ip, + } } +} diff --git a/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/templates/userdata.tftpl b/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/templates/userdata.tftpl new file mode 100644 index 0000000..55e626a --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/templates/userdata.tftpl @@ -0,0 +1,45 @@ +#!/bin/bash + +################################################# +## +## IP Address +## +configure_private_ip_address () { + cat << EOS > /etc/netplan/01-netcfg.yaml +network: + version: 2 + renderer: networkd + ethernets: + ens192: + dhcp4: yes + dhcp6: yes + dhcp-identifier: mac + ens224: + dhcp4: no + dhcp6: no + addresses: [${private_ip_address}] +EOS + netplan apply +} +configure_private_ip_address + +################################################# +## +## SSH +## +configure_ssh_port () { + sed -i 's/^#*Port [0-9]*/Port ${ssh_port}/' /etc/ssh/sshd_config +} +configure_ssh_port + +################################################# +## +## Hostname +## +hostnamectl set-hostname ${hostname} + +################################################# +## +## Disable swap files genereated by systemd-gpt-auto-generator +## +systemctl mask "dev-sda3.swap" diff --git a/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/terraform.tf b/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/terraform.tf new file mode 100644 index 0000000..97ef484 --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/terraform.tf @@ -0,0 +1,9 @@ +terraform { + required_version = ">=1.3.7" + required_providers { + nifcloud = { + source = "nifcloud/nifcloud" + version = ">= 1.8.0, < 2.0.0" + } + } +} diff --git a/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/variables.tf b/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/variables.tf new file mode 100644 index 0000000..65c11fe --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/modules/kubernetes-cluster/variables.tf @@ -0,0 +1,81 @@ +variable "availability_zone" { + description = "The availability zone" + type = string +} + +variable "prefix" { + description = "The prefix for the entire cluster" + type = string + validation { + condition = length(var.prefix) <= 5 + error_message = "Must be a less than 5 character long." + } +} + +variable "private_network_cidr" { + description = "The subnet of private network" + type = string + validation { + condition = can(cidrnetmask(var.private_network_cidr)) + error_message = "Must be a valid IPv4 CIDR block address." + } +} + +variable "private_ip_bn" { + description = "Private IP of bastion server" + type = string +} + +variable "instances_cp" { + type = map(object({ + private_ip = string + })) +} + +variable "instances_wk" { + type = map(object({ + private_ip = string + })) +} + +variable "instance_key_name" { + description = "The key name of the Key Pair to use for the instance" + type = string +} + +variable "instance_type_bn" { + description = "The instance type of bastion server" + type = string +} + +variable "instance_type_wk" { + description = "The instance type of worker" + type = string +} + +variable "instance_type_cp" { + description = "The instance type of control plane" + type = string +} + +variable "image_name" { + description = "The name of image" + type = string +} + +variable "additional_lb_filter" { + description = "Additional LB filter" + type = list(string) +} + +variable "accounting_type" { + type = string + default = "1" + validation { + condition = anytrue([ + var.accounting_type == "1", // Monthly + var.accounting_type == "2", // Pay per use + ]) + error_message = "Must be a 1 or 2." + } +} diff --git a/kubespray/contrib/terraform/nifcloud/output.tf b/kubespray/contrib/terraform/nifcloud/output.tf new file mode 100644 index 0000000..dcdeacb --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/output.tf @@ -0,0 +1,3 @@ +output "kubernetes_cluster" { + value = module.kubernetes_cluster +} diff --git a/kubespray/contrib/terraform/nifcloud/sample-inventory/cluster.tfvars b/kubespray/contrib/terraform/nifcloud/sample-inventory/cluster.tfvars new file mode 100644 index 0000000..3410a54 --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/sample-inventory/cluster.tfvars @@ -0,0 +1,22 @@ +region = "jp-west-1" +az = "west-11" + +instance_key_name = "deployerkey" + +instance_type_bn = "e-medium" +instance_type_cp = "e-medium" +instance_type_wk = "e-medium" + +private_network_cidr = "192.168.30.0/24" +instances_cp = { + "cp01" : { private_ip : "192.168.30.11/24" } + "cp02" : { private_ip : "192.168.30.12/24" } + "cp03" : { private_ip : "192.168.30.13/24" } +} +instances_wk = { + "wk01" : { private_ip : "192.168.30.21/24" } + "wk02" : { private_ip : "192.168.30.22/24" } +} +private_ip_bn = "192.168.30.10/24" + +image_name = "Ubuntu Server 22.04 LTS" diff --git a/kubespray/contrib/terraform/nifcloud/sample-inventory/group_vars b/kubespray/contrib/terraform/nifcloud/sample-inventory/group_vars new file mode 120000 index 0000000..3735958 --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/sample-inventory/group_vars @@ -0,0 +1 @@ +../../../../inventory/sample/group_vars \ No newline at end of file diff --git a/kubespray/contrib/terraform/nifcloud/terraform.tf b/kubespray/contrib/terraform/nifcloud/terraform.tf new file mode 100644 index 0000000..9a14bc6 --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/terraform.tf @@ -0,0 +1,9 @@ +terraform { + required_version = ">=1.3.7" + required_providers { + nifcloud = { + source = "nifcloud/nifcloud" + version = "1.8.0" + } + } +} diff --git a/kubespray/contrib/terraform/nifcloud/variables.tf b/kubespray/contrib/terraform/nifcloud/variables.tf new file mode 100644 index 0000000..558655f --- /dev/null +++ b/kubespray/contrib/terraform/nifcloud/variables.tf @@ -0,0 +1,77 @@ +variable "region" { + description = "The region" + type = string +} + +variable "az" { + description = "The availability zone" + type = string +} + +variable "private_ip_bn" { + description = "Private IP of bastion server" + type = string +} + +variable "private_network_cidr" { + description = "The subnet of private network" + type = string + validation { + condition = can(cidrnetmask(var.private_network_cidr)) + error_message = "Must be a valid IPv4 CIDR block address." + } +} + +variable "instances_cp" { + type = map(object({ + private_ip = string + })) +} + +variable "instances_wk" { + type = map(object({ + private_ip = string + })) +} + +variable "instance_key_name" { + description = "The key name of the Key Pair to use for the instance" + type = string +} + +variable "instance_type_bn" { + description = "The instance type of bastion server" + type = string +} + +variable "instance_type_wk" { + description = "The instance type of worker" + type = string +} + +variable "instance_type_cp" { + description = "The instance type of control plane" + type = string +} + +variable "image_name" { + description = "The name of image" + type = string +} + +variable "working_instance_ip" { + description = "The IP address to connect to bastion server." + type = string +} + +variable "accounting_type" { + type = string + default = "2" + validation { + condition = anytrue([ + var.accounting_type == "1", // Monthly + var.accounting_type == "2", // Pay per use + ]) + error_message = "Must be a 1 or 2." + } +} diff --git a/kubespray/contrib/terraform/openstack/.gitignore b/kubespray/contrib/terraform/openstack/.gitignore new file mode 100644 index 0000000..55d775b --- /dev/null +++ b/kubespray/contrib/terraform/openstack/.gitignore @@ -0,0 +1,5 @@ +.terraform +*.tfvars +!sample-inventory\/cluster.tfvars +*.tfstate +*.tfstate.backup diff --git a/kubespray/contrib/terraform/openstack/README.md b/kubespray/contrib/terraform/openstack/README.md new file mode 100644 index 0000000..a996692 --- /dev/null +++ b/kubespray/contrib/terraform/openstack/README.md @@ -0,0 +1,786 @@ +# Kubernetes on OpenStack with Terraform + +Provision a Kubernetes cluster with [Terraform](https://www.terraform.io) on +OpenStack. + +## Status + +This will install a Kubernetes cluster on an OpenStack Cloud. It should work on +most modern installs of OpenStack that support the basic services. + +### Known compatible public clouds + +- [Auro](https://auro.io/) +- [Betacloud](https://www.betacloud.io/) +- [CityCloud](https://www.citycloud.com/) +- [DreamHost](https://www.dreamhost.com/cloud/computing/) +- [ELASTX](https://elastx.se/) +- [EnterCloudSuite](https://www.entercloudsuite.com/) +- [FugaCloud](https://fuga.cloud/) +- [Open Telekom Cloud](https://cloud.telekom.de/) +- [OVH](https://www.ovh.com/) +- [Rackspace](https://www.rackspace.com/) +- [Safespring](https://www.safespring.com) +- [Ultimum](https://ultimum.io/) +- [VexxHost](https://vexxhost.com/) +- [Zetta](https://www.zetta.io/) + +## Approach + +The terraform configuration inspects variables found in +[variables.tf](variables.tf) to create resources in your OpenStack cluster. +There is a [python script](../terraform.py) that reads the generated`.tfstate` +file to generate a dynamic inventory that is consumed by the main ansible script +to actually install kubernetes and stand up the cluster. + +### Networking + +The configuration includes creating a private subnet with a router to the +external net. It will allocate floating IPs from a pool and assign them to the +hosts where that makes sense. You have the option of creating bastion hosts +inside the private subnet to access the nodes there. Alternatively, a node with +a floating IP can be used as a jump host to nodes without. + +#### Using an existing router + +It is possible to use an existing router instead of creating one. To use an +existing router set the router\_id variable to the uuid of the router you wish +to use. + +For example: + +```ShellSession +router_id = "00c542e7-6f46-4535-ae95-984c7f0391a3" +``` + +### Kubernetes Nodes + +You can create many different kubernetes topologies by setting the number of +different classes of hosts. For each class there are options for allocating +floating IP addresses or not. + +- Master nodes with etcd +- Master nodes without etcd +- Standalone etcd hosts +- Kubernetes worker nodes + +Note that the Ansible script will report an invalid configuration if you wind up +with an even number of etcd instances since that is not a valid configuration. This +restriction includes standalone etcd nodes that are deployed in a cluster along with +master nodes with etcd replicas. As an example, if you have three master nodes with +etcd replicas and three standalone etcd nodes, the script will fail since there are +now six total etcd replicas. + +### GlusterFS shared file system + +The Terraform configuration supports provisioning of an optional GlusterFS +shared file system based on a separate set of VMs. To enable this, you need to +specify: + +- the number of Gluster hosts (minimum 2) +- Size of the non-ephemeral volumes to be attached to store the GlusterFS bricks +- Other properties related to provisioning the hosts + +Even if you are using Flatcar Container Linux by Kinvolk for your cluster, you will still +need the GlusterFS VMs to be based on either Debian or RedHat based images. +Flatcar Container Linux by Kinvolk cannot serve GlusterFS, but can connect to it through +binaries available on hyperkube v1.4.3_coreos.0 or higher. + +## Requirements + +- [Install Terraform](https://www.terraform.io/intro/getting-started/install.html) 0.14 or later +- [Install Ansible](http://docs.ansible.com/ansible/latest/intro_installation.html) +- you already have a suitable OS image in Glance +- you already have a floating IP pool created +- you have security groups enabled +- you have a pair of keys generated that can be used to secure the new hosts + +## Module Architecture + +The configuration is divided into three modules: + +- Network +- IPs +- Compute + +The main reason for splitting the configuration up in this way is to easily +accommodate situations where floating IPs are limited by a quota or if you have +any external references to the floating IP (e.g. DNS) that would otherwise have +to be updated. + +You can force your existing IPs by modifying the compute variables in +`kubespray.tf` as follows: + +```ini +k8s_master_fips = ["151.101.129.67"] +k8s_node_fips = ["151.101.129.68"] +``` + +## Terraform + +Terraform will be used to provision all of the OpenStack resources with base software as appropriate. + +### Configuration + +#### Inventory files + +Create an inventory directory for your cluster by copying the existing sample and linking the `hosts` script (used to build the inventory based on Terraform state): + +```ShellSession +cp -LRp contrib/terraform/openstack/sample-inventory inventory/$CLUSTER +cd inventory/$CLUSTER +ln -s ../../contrib/terraform/openstack/hosts +ln -s ../../contrib +``` + +This will be the base for subsequent Terraform commands. + +#### OpenStack access and credentials + +No provider variables are hardcoded inside `variables.tf` because Terraform +supports various authentication methods for OpenStack: the older script and +environment method (using `openrc`) as well as a newer declarative method, and +different OpenStack environments may support Identity API version 2 or 3. + +These are examples and may vary depending on your OpenStack cloud provider, +for an exhaustive list on how to authenticate on OpenStack with Terraform +please read the [OpenStack provider documentation](https://www.terraform.io/docs/providers/openstack/). + +##### Declarative method (recommended) + +The recommended authentication method is to describe credentials in a YAML file `clouds.yaml` that can be stored in: + +- the current directory +- `~/.config/openstack` +- `/etc/openstack` + +`clouds.yaml`: + +```yaml +clouds: + mycloud: + auth: + auth_url: https://openstack:5000/v3 + username: "username" + project_name: "projectname" + project_id: projectid + user_domain_name: "Default" + password: "password" + region_name: "RegionOne" + interface: "public" + identity_api_version: 3 +``` + +If you have multiple clouds defined in your `clouds.yaml` file you can choose +the one you want to use with the environment variable `OS_CLOUD`: + +```ShellSession +export OS_CLOUD=mycloud +``` + +##### Openrc method + +When using classic environment variables, Terraform uses default `OS_*` +environment variables. A script suitable for your environment may be available +from Horizon under *Project* -> *Compute* -> *Access & Security* -> *API Access*. + +With identity v2: + +```ShellSession +source openrc + +env | grep OS + +OS_AUTH_URL=https://openstack:5000/v2.0 +OS_PROJECT_ID=projectid +OS_PROJECT_NAME=projectname +OS_USERNAME=username +OS_PASSWORD=password +OS_REGION_NAME=RegionOne +OS_INTERFACE=public +OS_IDENTITY_API_VERSION=2 +``` + +With identity v3: + +```ShellSession +source openrc + +env | grep OS + +OS_AUTH_URL=https://openstack:5000/v3 +OS_PROJECT_ID=projectid +OS_PROJECT_NAME=username +OS_PROJECT_DOMAIN_ID=default +OS_USERNAME=username +OS_PASSWORD=password +OS_REGION_NAME=RegionOne +OS_INTERFACE=public +OS_IDENTITY_API_VERSION=3 +OS_USER_DOMAIN_NAME=Default +``` + +Terraform does not support a mix of DomainName and DomainID, choose one or the other: + +- provider.openstack: You must provide exactly one of DomainID or DomainName to authenticate by Username + +```ShellSession +unset OS_USER_DOMAIN_NAME +export OS_USER_DOMAIN_ID=default +``` + +or + +```ShellSession +unset OS_PROJECT_DOMAIN_ID +set OS_PROJECT_DOMAIN_NAME=Default +``` + +#### Cluster variables + +The construction of the cluster is driven by values found in +[variables.tf](variables.tf). + +For your cluster, edit `inventory/$CLUSTER/cluster.tfvars`. + +|Variable | Description | +|---------|-------------| +|`cluster_name` | All OpenStack resources will use the Terraform variable`cluster_name` (default`example`) in their name to make it easier to track. For example the first compute resource will be named`example-kubernetes-1`. | +|`az_list` | List of Availability Zones available in your OpenStack cluster. | +|`network_name` | The name to be given to the internal network that will be generated | +|`use_existing_network`| Use an existing network with the name of `network_name`. `false` by default | +|`network_dns_domain` | (Optional) The dns_domain for the internal network that will be generated | +|`dns_nameservers`| An array of DNS name server names to be used by hosts in the internal subnet. | +|`floatingip_pool` | Name of the pool from which floating IPs will be allocated | +|`k8s_master_fips` | A list of floating IPs that you have already pre-allocated; they will be attached to master nodes instead of creating new random floating IPs. | +|`bastion_fips` | A list of floating IPs that you have already pre-allocated; they will be attached to bastion node instead of creating new random floating IPs. | +|`external_net` | UUID of the external network that will be routed to | +|`flavor_k8s_master`,`flavor_k8s_node`,`flavor_etcd`, `flavor_bastion`,`flavor_gfs_node` | Flavor depends on your openstack installation, you can get available flavor IDs through `openstack flavor list` | +|`image`,`image_gfs` | Name of the image to use in provisioning the compute resources. Should already be loaded into glance. | +|`ssh_user`,`ssh_user_gfs` | The username to ssh into the image with. This usually depends on the image you have selected | +|`public_key_path` | Path on your local workstation to the public key file you wish to use in creating the key pairs | +|`number_of_k8s_masters`, `number_of_k8s_masters_no_floating_ip` | Number of nodes that serve as both master and etcd. These can be provisioned with or without floating IP addresses| +|`number_of_k8s_masters_no_etcd`, `number_of_k8s_masters_no_floating_ip_no_etcd` | Number of nodes that serve as just master with no etcd. These can be provisioned with or without floating IP addresses | +|`number_of_etcd` | Number of pure etcd nodes | +|`number_of_k8s_nodes`, `number_of_k8s_nodes_no_floating_ip` | Kubernetes worker nodes. These can be provisioned with or without floating ip addresses. | +|`number_of_bastions` | Number of bastion hosts to create. Scripts assume this is really just zero or one | +|`number_of_gfs_nodes_no_floating_ip` | Number of gluster servers to provision. | +| `gfs_volume_size_in_gb` | Size of the non-ephemeral volumes to be attached to store the GlusterFS bricks | +|`supplementary_master_groups` | To add ansible groups to the masters, such as `kube_node` for tainting them as nodes, empty by default. | +|`supplementary_node_groups` | To add ansible groups to the nodes, such as `kube_ingress` for running ingress controller pods, empty by default. | +|`bastion_allowed_remote_ips` | List of CIDR allowed to initiate a SSH connection, `["0.0.0.0/0"]` by default | +|`master_allowed_remote_ips` | List of CIDR blocks allowed to initiate an API connection, `["0.0.0.0/0"]` by default | +|`bastion_allowed_ports` | List of ports to open on bastion node, `[]` by default | +|`k8s_allowed_remote_ips` | List of CIDR allowed to initiate a SSH connection, empty by default | +|`worker_allowed_ports` | List of ports to open on worker nodes, `[{ "protocol" = "tcp", "port_range_min" = 30000, "port_range_max" = 32767, "remote_ip_prefix" = "0.0.0.0/0"}]` by default | +|`master_allowed_ports` | List of ports to open on master nodes, expected format is `[{ "protocol" = "tcp", "port_range_min" = 443, "port_range_max" = 443, "remote_ip_prefix" = "0.0.0.0/0"}]`, empty by default | +|`node_root_volume_size_in_gb` | Size of the root volume for nodes, 0 to use ephemeral storage | +|`master_root_volume_size_in_gb` | Size of the root volume for masters, 0 to use ephemeral storage | +|`master_volume_type` | Volume type of the root volume for control_plane, 'Default' by default | +|`node_volume_type` | Volume type of the root volume for nodes, 'Default' by default | +|`gfs_root_volume_size_in_gb` | Size of the root volume for gluster, 0 to use ephemeral storage | +|`etcd_root_volume_size_in_gb` | Size of the root volume for etcd nodes, 0 to use ephemeral storage | +|`bastion_root_volume_size_in_gb` | Size of the root volume for bastions, 0 to use ephemeral storage | +|`master_server_group_policy` | Enable and use openstack nova servergroups for masters with set policy, default: "" (disabled) | +|`node_server_group_policy` | Enable and use openstack nova servergroups for nodes with set policy, default: "" (disabled) | +|`etcd_server_group_policy` | Enable and use openstack nova servergroups for etcd with set policy, default: "" (disabled) | +|`additional_server_groups` | Extra server groups to create. Set "policy" to the policy for the group, expected format is `{"new-server-group" = {"policy" = "anti-affinity"}}`, default: {} (to not create any extra groups) | +|`use_access_ip` | If 1, nodes with floating IPs will transmit internal cluster traffic via floating IPs; if 0 private IPs will be used instead. Default value is 1. | +|`port_security_enabled` | Allow to disable port security by setting this to `false`. `true` by default | +|`force_null_port_security` | Set `null` instead of `true` or `false` for `port_security`. `false` by default | +|`k8s_nodes` | Map containing worker node definition, see explanation below | +|`k8s_masters` | Map containing master node definition, see explanation for k8s_nodes and `sample-inventory/cluster.tfvars` | + +##### k8s_nodes + +Allows a custom definition of worker nodes giving the operator full control over individual node flavor and availability zone placement. +To enable the use of this mode set the `number_of_k8s_nodes` and `number_of_k8s_nodes_no_floating_ip` variables to 0. +Then define your desired worker node configuration using the `k8s_nodes` variable. +The `az`, `flavor` and `floating_ip` parameters are mandatory. +The optional parameter `extra_groups` (a comma-delimited string) can be used to define extra inventory group memberships for specific nodes. + +```yaml +k8s_nodes: + node-name: + az: string # Name of the AZ + flavor: string # Flavor ID to use + floating_ip: bool # If floating IPs should be created or not + extra_groups: string # (optional) Additional groups to add for kubespray, defaults to no groups + image_id: string # (optional) Image ID to use, defaults to var.image_id or var.image + root_volume_size_in_gb: number # (optional) Size of the block storage to use as root disk, defaults to var.node_root_volume_size_in_gb or to use volume from flavor otherwise + volume_type: string # (optional) Volume type to use, defaults to var.node_volume_type + network_id: string # (optional) Use this network_id for the node, defaults to either var.network_id or ID of var.network_name + server_group: string # (optional) Server group to add this node to. If set, this has to be one specified in additional_server_groups, defaults to use the server group specified in node_server_group_policy + cloudinit: # (optional) Options for cloud-init + extra_partitions: # List of extra partitions (other than the root partition) to setup during creation + volume_path: string # Path to the volume to create partition for (e.g. /dev/vda ) + partition_path: string # Path to the partition (e.g. /dev/vda2 ) + mount_path: string # Path to where the partition should be mounted + partition_start: string # Where the partition should start (e.g. 10GB ). Note, if you set the partition_start to 0 there will be no space left for the root partition + partition_end: string # Where the partition should end (e.g. 10GB or -1 for end of volume) + netplan_critical_dhcp_interface: string # Name of interface to set the dhcp flag critical = true, to circumvent [this issue](https://bugs.launchpad.net/ubuntu/+source/systemd/+bug/1776013). +``` + +For example: + +```ini +k8s_nodes = { + "1" = { + "az" = "sto1" + "flavor" = "83d8b44a-26a0-4f02-a981-079446926445" + "floating_ip" = true + }, + "2" = { + "az" = "sto2" + "flavor" = "83d8b44a-26a0-4f02-a981-079446926445" + "floating_ip" = true + }, + "3" = { + "az" = "sto3" + "flavor" = "83d8b44a-26a0-4f02-a981-079446926445" + "floating_ip" = true + "extra_groups" = "calico_rr" + } +} +``` + +Would result in the same configuration as: + +```ini +number_of_k8s_nodes = 3 +flavor_k8s_node = "83d8b44a-26a0-4f02-a981-079446926445" +az_list = ["sto1", "sto2", "sto3"] +``` + +And: + +```ini +k8s_nodes = { + "ing-1" = { + "az" = "sto1" + "flavor" = "83d8b44a-26a0-4f02-a981-079446926445" + "floating_ip" = true + }, + "ing-2" = { + "az" = "sto2" + "flavor" = "83d8b44a-26a0-4f02-a981-079446926445" + "floating_ip" = true + }, + "ing-3" = { + "az" = "sto3" + "flavor" = "83d8b44a-26a0-4f02-a981-079446926445" + "floating_ip" = true + }, + "big-1" = { + "az" = "sto1" + "flavor" = "3f73fc93-ec61-4808-88df-2580d94c1a9b" + "floating_ip" = false + }, + "big-2" = { + "az" = "sto2" + "flavor" = "3f73fc93-ec61-4808-88df-2580d94c1a9b" + "floating_ip" = false + }, + "big-3" = { + "az" = "sto3" + "flavor" = "3f73fc93-ec61-4808-88df-2580d94c1a9b" + "floating_ip" = false + }, + "small-1" = { + "az" = "sto1" + "flavor" = "7a6a998f-ac7f-4fb8-a534-2175b254f75e" + "floating_ip" = false + }, + "small-2" = { + "az" = "sto2" + "flavor" = "7a6a998f-ac7f-4fb8-a534-2175b254f75e" + "floating_ip" = false + }, + "small-3" = { + "az" = "sto3" + "flavor" = "7a6a998f-ac7f-4fb8-a534-2175b254f75e" + "floating_ip" = false + } +} +``` + +Would result in three nodes in each availability zone each with their own separate naming, +flavor and floating ip configuration. + +The "schema": + +```ini +k8s_nodes = { + "key | node name suffix, must be unique" = { + "az" = string + "flavor" = string + "floating_ip" = bool + }, +} +``` + +All values are required. + +#### Terraform state files + +In the cluster's inventory folder, the following files might be created (either by Terraform +or manually), to prevent you from pushing them accidentally they are in a +`.gitignore` file in the `terraform/openstack` directory : + +- `.terraform` +- `.tfvars` +- `.tfstate` +- `.tfstate.backup` + +You can still add them manually if you want to. + +### Initialization + +Before Terraform can operate on your cluster you need to install the required +plugins. This is accomplished as follows: + +```ShellSession +cd inventory/$CLUSTER +terraform -chdir="../../contrib/terraform/openstack" init +``` + +This should finish fairly quickly telling you Terraform has successfully initialized and loaded necessary modules. + +### Customizing with cloud-init + +You can apply cloud-init based customization for the openstack instances before provisioning your cluster. +One common template is used for all instances. Adjust the file shown below: +`contrib/terraform/openstack/modules/compute/templates/cloudinit.yaml.tmpl` +For example, to enable openstack novnc access and ansible_user=root SSH access: + +```ShellSession +#cloud-config +## in some cases novnc console access is required +## it requires ssh password to be set +ssh_pwauth: yes +chpasswd: + list: | + root:secret + expire: False + +## in some cases direct root ssh access via ssh key is required +disable_root: false +``` + +### Provisioning cluster + +You can apply the Terraform configuration to your cluster with the following command +issued from your cluster's inventory directory (`inventory/$CLUSTER`): + +```ShellSession +terraform -chdir="../../contrib/terraform/openstack" apply -var-file=cluster.tfvars +``` + +if you chose to create a bastion host, this script will create +`contrib/terraform/openstack/k8s_cluster.yml` with an ssh command for Ansible to +be able to access your machines tunneling through the bastion's IP address. If +you want to manually handle the ssh tunneling to these machines, please delete +or move that file. If you want to use this, just leave it there, as ansible will +pick it up automatically. + +### Destroying cluster + +You can destroy your new cluster with the following command issued from the cluster's inventory directory: + +```ShellSession +terraform -chdir="../../contrib/terraform/openstack" destroy -var-file=cluster.tfvars +``` + +If you've started the Ansible run, it may also be a good idea to do some manual cleanup: + +- remove SSH keys from the destroyed cluster from your `~/.ssh/known_hosts` file +- clean up any temporary cache files: `rm /tmp/$CLUSTER-*` + +### Debugging + +You can enable debugging output from Terraform by setting +`OS_DEBUG` to 1 and`TF_LOG` to`DEBUG` before running the Terraform command. + +### Terraform output + +Terraform can output values that are useful for configure Neutron/Octavia LBaaS or Cinder persistent volume provisioning as part of your Kubernetes deployment: + +- `private_subnet_id`: the subnet where your instances are running is used for `openstack_lbaas_subnet_id` +- `floating_network_id`: the network_id where the floating IP are provisioned is used for `openstack_lbaas_floating_network_id` + +## Ansible + +### Node access + +#### SSH + +Ensure your local ssh-agent is running and your ssh key has been added. This +step is required by the terraform provisioner: + +```ShellSession +eval $(ssh-agent -s) +ssh-add ~/.ssh/id_rsa +``` + +If you have deployed and destroyed a previous iteration of your cluster, you will need to clear out any stale keys from your SSH "known hosts" file ( `~/.ssh/known_hosts`). + +#### Metadata variables + +The [python script](../terraform.py) that reads the +generated`.tfstate` file to generate a dynamic inventory recognizes +some variables within a "metadata" block, defined in a "resource" +block (example): + +```ini +resource "openstack_compute_instance_v2" "example" { + ... + metadata { + ssh_user = "ubuntu" + prefer_ipv6 = true + python_bin = "/usr/bin/python3" + } + ... +} +``` + +As the example shows, these let you define the SSH username for +Ansible, a Python binary which is needed by Ansible if +`/usr/bin/python` doesn't exist, and whether the IPv6 address of the +instance should be preferred over IPv4. + +#### Bastion host + +Bastion access will be determined by: + +- Your choice on the amount of bastion hosts (set by `number_of_bastions` terraform variable). +- The existence of nodes/masters with floating IPs (set by `number_of_k8s_masters`, `number_of_k8s_nodes`, `number_of_k8s_masters_no_etcd` terraform variables). + +If you have a bastion host, your ssh traffic will be directly routed through it. This is regardless of whether you have masters/nodes with a floating IP assigned. +If you don't have a bastion host, but at least one of your masters/nodes have a floating IP, then ssh traffic will be tunneled by one of these machines. + +So, either a bastion host, or at least master/node with a floating IP are required. + +#### Test access + +Make sure you can connect to the hosts. Note that Flatcar Container Linux by Kinvolk will have a state `FAILED` due to Python not being present. This is okay, because Python will be installed during bootstrapping, so long as the hosts are not `UNREACHABLE`. + +```ShellSession +$ ansible -i inventory/$CLUSTER/hosts -m ping all +example-k8s_node-1 | SUCCESS => { + "changed": false, + "ping": "pong" +} +example-etcd-1 | SUCCESS => { + "changed": false, + "ping": "pong" +} +example-k8s-master-1 | SUCCESS => { + "changed": false, + "ping": "pong" +} +``` + +If it fails try to connect manually via SSH. It could be something as simple as a stale host key. + +### Configure cluster variables + +Edit `inventory/$CLUSTER/group_vars/all/all.yml`: + +- **bin_dir**: + +```yml +# Directory where the binaries will be installed +# Default: +# bin_dir: /usr/local/bin +# For Flatcar Container Linux by Kinvolk: +bin_dir: /opt/bin +``` + +- and **cloud_provider**: + +```yml +cloud_provider: openstack +``` + +Edit `inventory/$CLUSTER/group_vars/k8s_cluster/k8s_cluster.yml`: + +- Set variable **kube_network_plugin** to your desired networking plugin. + - **flannel** works out-of-the-box + - **calico** requires [configuring OpenStack Neutron ports](/docs/openstack.md) to allow service and pod subnets + +```yml +# Choose network plugin (calico, weave or flannel) +# Can also be set to 'cloud', which lets the cloud provider setup appropriate routing +kube_network_plugin: flannel +``` + +- Set variable **resolvconf_mode** + +```yml +# Can be docker_dns, host_resolvconf or none +# Default: +# resolvconf_mode: docker_dns +# For Flatcar Container Linux by Kinvolk: +resolvconf_mode: host_resolvconf +``` + +- Set max amount of attached cinder volume per host (default 256) + +```yml +node_volume_attach_limit: 26 +``` + +### Deploy Kubernetes + +```ShellSession +ansible-playbook --become -i inventory/$CLUSTER/hosts cluster.yml +``` + +This will take some time as there are many tasks to run. + +## Kubernetes + +### Set up kubectl + +1. [Install kubectl](https://kubernetes.io/docs/tasks/tools/install-kubectl/) on your workstation +2. Add a route to the internal IP of a master node (if needed): + +```ShellSession +sudo route add [master-internal-ip] gw [router-ip] +``` + +or + +```ShellSession +sudo route add -net [internal-subnet]/24 gw [router-ip] +``` + +1. List Kubernetes certificates & keys: + +```ShellSession +ssh [os-user]@[master-ip] sudo ls /etc/kubernetes/ssl/ +``` + +1. Get `admin`'s certificates and keys: + +```ShellSession +ssh [os-user]@[master-ip] sudo cat /etc/kubernetes/ssl/admin-kube-master-1-key.pem > admin-key.pem +ssh [os-user]@[master-ip] sudo cat /etc/kubernetes/ssl/admin-kube-master-1.pem > admin.pem +ssh [os-user]@[master-ip] sudo cat /etc/kubernetes/ssl/ca.pem > ca.pem +``` + +1. Configure kubectl: + +```ShellSession +$ kubectl config set-cluster default-cluster --server=https://[master-internal-ip]:6443 \ + --certificate-authority=ca.pem + +$ kubectl config set-credentials default-admin \ + --certificate-authority=ca.pem \ + --client-key=admin-key.pem \ + --client-certificate=admin.pem + +$ kubectl config set-context default-system --cluster=default-cluster --user=default-admin +$ kubectl config use-context default-system +``` + +1. Check it: + +```ShellSession +kubectl version +``` + +## GlusterFS + +GlusterFS is not deployed by the standard `cluster.yml` playbook, see the +[GlusterFS playbook documentation](../../network-storage/glusterfs/README.md) +for instructions. + +Basically you will install Gluster as + +```ShellSession +ansible-playbook --become -i inventory/$CLUSTER/hosts ./contrib/network-storage/glusterfs/glusterfs.yml +``` + +## What's next + +Try out your new Kubernetes cluster with the [Hello Kubernetes service](https://kubernetes.io/docs/tasks/access-application-cluster/service-access-application-cluster/). + +## Appendix + +### Migration from `number_of_k8s_nodes*` to `k8s_nodes` + +If you currently have a cluster defined using the `number_of_k8s_nodes*` variables and wish +to migrate to the `k8s_nodes` style you can do it like so: + +```ShellSession +$ terraform state list +module.compute.data.openstack_images_image_v2.gfs_image +module.compute.data.openstack_images_image_v2.vm_image +module.compute.openstack_compute_floatingip_associate_v2.k8s_master[0] +module.compute.openstack_compute_floatingip_associate_v2.k8s_node[0] +module.compute.openstack_compute_floatingip_associate_v2.k8s_node[1] +module.compute.openstack_compute_floatingip_associate_v2.k8s_node[2] +module.compute.openstack_compute_instance_v2.k8s_master[0] +module.compute.openstack_compute_instance_v2.k8s_node[0] +module.compute.openstack_compute_instance_v2.k8s_node[1] +module.compute.openstack_compute_instance_v2.k8s_node[2] +module.compute.openstack_compute_keypair_v2.k8s +module.compute.openstack_compute_servergroup_v2.k8s_etcd[0] +module.compute.openstack_compute_servergroup_v2.k8s_master[0] +module.compute.openstack_compute_servergroup_v2.k8s_node[0] +module.compute.openstack_networking_secgroup_rule_v2.bastion[0] +module.compute.openstack_networking_secgroup_rule_v2.egress[0] +module.compute.openstack_networking_secgroup_rule_v2.k8s +module.compute.openstack_networking_secgroup_rule_v2.k8s_allowed_remote_ips[0] +module.compute.openstack_networking_secgroup_rule_v2.k8s_allowed_remote_ips[1] +module.compute.openstack_networking_secgroup_rule_v2.k8s_allowed_remote_ips[2] +module.compute.openstack_networking_secgroup_rule_v2.k8s_master[0] +module.compute.openstack_networking_secgroup_rule_v2.worker[0] +module.compute.openstack_networking_secgroup_rule_v2.worker[1] +module.compute.openstack_networking_secgroup_rule_v2.worker[2] +module.compute.openstack_networking_secgroup_rule_v2.worker[3] +module.compute.openstack_networking_secgroup_rule_v2.worker[4] +module.compute.openstack_networking_secgroup_v2.bastion[0] +module.compute.openstack_networking_secgroup_v2.k8s +module.compute.openstack_networking_secgroup_v2.k8s_master +module.compute.openstack_networking_secgroup_v2.worker +module.ips.null_resource.dummy_dependency +module.ips.openstack_networking_floatingip_v2.k8s_master[0] +module.ips.openstack_networking_floatingip_v2.k8s_node[0] +module.ips.openstack_networking_floatingip_v2.k8s_node[1] +module.ips.openstack_networking_floatingip_v2.k8s_node[2] +module.network.openstack_networking_network_v2.k8s[0] +module.network.openstack_networking_router_interface_v2.k8s[0] +module.network.openstack_networking_router_v2.k8s[0] +module.network.openstack_networking_subnet_v2.k8s[0] +$ terraform state mv 'module.compute.openstack_compute_floatingip_associate_v2.k8s_node[0]' 'module.compute.openstack_compute_floatingip_associate_v2.k8s_nodes["1"]' +Move "module.compute.openstack_compute_floatingip_associate_v2.k8s_node[0]" to "module.compute.openstack_compute_floatingip_associate_v2.k8s_nodes[\"1\"]" +Successfully moved 1 object(s). +$ terraform state mv 'module.compute.openstack_compute_floatingip_associate_v2.k8s_node[1]' 'module.compute.openstack_compute_floatingip_associate_v2.k8s_nodes["2"]' +Move "module.compute.openstack_compute_floatingip_associate_v2.k8s_node[1]" to "module.compute.openstack_compute_floatingip_associate_v2.k8s_nodes[\"2\"]" +Successfully moved 1 object(s). +$ terraform state mv 'module.compute.openstack_compute_floatingip_associate_v2.k8s_node[2]' 'module.compute.openstack_compute_floatingip_associate_v2.k8s_nodes["3"]' +Move "module.compute.openstack_compute_floatingip_associate_v2.k8s_node[2]" to "module.compute.openstack_compute_floatingip_associate_v2.k8s_nodes[\"3\"]" +Successfully moved 1 object(s). +$ terraform state mv 'module.compute.openstack_compute_instance_v2.k8s_node[0]' 'module.compute.openstack_compute_instance_v2.k8s_node["1"]' +Move "module.compute.openstack_compute_instance_v2.k8s_node[0]" to "module.compute.openstack_compute_instance_v2.k8s_node[\"1\"]" +Successfully moved 1 object(s). +$ terraform state mv 'module.compute.openstack_compute_instance_v2.k8s_node[1]' 'module.compute.openstack_compute_instance_v2.k8s_node["2"]' +Move "module.compute.openstack_compute_instance_v2.k8s_node[1]" to "module.compute.openstack_compute_instance_v2.k8s_node[\"2\"]" +Successfully moved 1 object(s). +$ terraform state mv 'module.compute.openstack_compute_instance_v2.k8s_node[2]' 'module.compute.openstack_compute_instance_v2.k8s_node["3"]' +Move "module.compute.openstack_compute_instance_v2.k8s_node[2]" to "module.compute.openstack_compute_instance_v2.k8s_node[\"3\"]" +Successfully moved 1 object(s). +$ terraform state mv 'module.ips.openstack_networking_floatingip_v2.k8s_node[0]' 'module.ips.openstack_networking_floatingip_v2.k8s_node["1"]' +Move "module.ips.openstack_networking_floatingip_v2.k8s_node[0]" to "module.ips.openstack_networking_floatingip_v2.k8s_node[\"1\"]" +Successfully moved 1 object(s). +$ terraform state mv 'module.ips.openstack_networking_floatingip_v2.k8s_node[1]' 'module.ips.openstack_networking_floatingip_v2.k8s_node["2"]' +Move "module.ips.openstack_networking_floatingip_v2.k8s_node[1]" to "module.ips.openstack_networking_floatingip_v2.k8s_node[\"2\"]" +Successfully moved 1 object(s). +$ terraform state mv 'module.ips.openstack_networking_floatingip_v2.k8s_node[2]' 'module.ips.openstack_networking_floatingip_v2.k8s_node["3"]' +Move "module.ips.openstack_networking_floatingip_v2.k8s_node[2]" to "module.ips.openstack_networking_floatingip_v2.k8s_node[\"3\"]" +Successfully moved 1 object(s). +``` + +Of course for nodes without floating ips those steps can be omitted. diff --git a/kubespray/contrib/terraform/openstack/hosts b/kubespray/contrib/terraform/openstack/hosts new file mode 120000 index 0000000..804b6fa --- /dev/null +++ b/kubespray/contrib/terraform/openstack/hosts @@ -0,0 +1 @@ +../terraform.py \ No newline at end of file diff --git a/kubespray/contrib/terraform/openstack/kubespray.tf b/kubespray/contrib/terraform/openstack/kubespray.tf new file mode 100644 index 0000000..a177634 --- /dev/null +++ b/kubespray/contrib/terraform/openstack/kubespray.tf @@ -0,0 +1,130 @@ +module "network" { + source = "./modules/network" + + external_net = var.external_net + network_name = var.network_name + subnet_cidr = var.subnet_cidr + cluster_name = var.cluster_name + dns_nameservers = var.dns_nameservers + network_dns_domain = var.network_dns_domain + use_neutron = var.use_neutron + port_security_enabled = var.port_security_enabled + router_id = var.router_id +} + +module "ips" { + source = "./modules/ips" + + number_of_k8s_masters = var.number_of_k8s_masters + number_of_k8s_masters_no_etcd = var.number_of_k8s_masters_no_etcd + number_of_k8s_nodes = var.number_of_k8s_nodes + floatingip_pool = var.floatingip_pool + number_of_bastions = var.number_of_bastions + external_net = var.external_net + network_name = var.network_name + router_id = module.network.router_id + k8s_nodes = var.k8s_nodes + k8s_masters = var.k8s_masters + k8s_master_fips = var.k8s_master_fips + bastion_fips = var.bastion_fips + router_internal_port_id = module.network.router_internal_port_id +} + +module "compute" { + source = "./modules/compute" + + cluster_name = var.cluster_name + az_list = var.az_list + az_list_node = var.az_list_node + number_of_k8s_masters = var.number_of_k8s_masters + number_of_k8s_masters_no_etcd = var.number_of_k8s_masters_no_etcd + number_of_etcd = var.number_of_etcd + number_of_k8s_masters_no_floating_ip = var.number_of_k8s_masters_no_floating_ip + number_of_k8s_masters_no_floating_ip_no_etcd = var.number_of_k8s_masters_no_floating_ip_no_etcd + number_of_k8s_nodes = var.number_of_k8s_nodes + number_of_bastions = var.number_of_bastions + number_of_k8s_nodes_no_floating_ip = var.number_of_k8s_nodes_no_floating_ip + number_of_gfs_nodes_no_floating_ip = var.number_of_gfs_nodes_no_floating_ip + k8s_masters = var.k8s_masters + k8s_nodes = var.k8s_nodes + bastion_root_volume_size_in_gb = var.bastion_root_volume_size_in_gb + etcd_root_volume_size_in_gb = var.etcd_root_volume_size_in_gb + master_root_volume_size_in_gb = var.master_root_volume_size_in_gb + node_root_volume_size_in_gb = var.node_root_volume_size_in_gb + gfs_root_volume_size_in_gb = var.gfs_root_volume_size_in_gb + gfs_volume_size_in_gb = var.gfs_volume_size_in_gb + master_volume_type = var.master_volume_type + node_volume_type = var.node_volume_type + public_key_path = var.public_key_path + image = var.image + image_uuid = var.image_uuid + image_gfs = var.image_gfs + image_master = var.image_master + image_master_uuid = var.image_master_uuid + image_gfs_uuid = var.image_gfs_uuid + ssh_user = var.ssh_user + ssh_user_gfs = var.ssh_user_gfs + flavor_k8s_master = var.flavor_k8s_master + flavor_k8s_node = var.flavor_k8s_node + flavor_etcd = var.flavor_etcd + flavor_gfs_node = var.flavor_gfs_node + network_name = var.network_name + flavor_bastion = var.flavor_bastion + k8s_master_fips = module.ips.k8s_master_fips + k8s_master_no_etcd_fips = module.ips.k8s_master_no_etcd_fips + k8s_masters_fips = module.ips.k8s_masters_fips + k8s_node_fips = module.ips.k8s_node_fips + k8s_nodes_fips = module.ips.k8s_nodes_fips + bastion_fips = module.ips.bastion_fips + bastion_allowed_remote_ips = var.bastion_allowed_remote_ips + master_allowed_remote_ips = var.master_allowed_remote_ips + k8s_allowed_remote_ips = var.k8s_allowed_remote_ips + k8s_allowed_egress_ips = var.k8s_allowed_egress_ips + supplementary_master_groups = var.supplementary_master_groups + supplementary_node_groups = var.supplementary_node_groups + master_allowed_ports = var.master_allowed_ports + worker_allowed_ports = var.worker_allowed_ports + bastion_allowed_ports = var.bastion_allowed_ports + use_access_ip = var.use_access_ip + master_server_group_policy = var.master_server_group_policy + node_server_group_policy = var.node_server_group_policy + etcd_server_group_policy = var.etcd_server_group_policy + extra_sec_groups = var.extra_sec_groups + extra_sec_groups_name = var.extra_sec_groups_name + group_vars_path = var.group_vars_path + port_security_enabled = var.port_security_enabled + force_null_port_security = var.force_null_port_security + network_router_id = module.network.router_id + network_id = module.network.network_id + use_existing_network = var.use_existing_network + private_subnet_id = module.network.subnet_id + additional_server_groups = var.additional_server_groups + + depends_on = [ + module.network.subnet_id + ] +} + +output "private_subnet_id" { + value = module.network.subnet_id +} + +output "floating_network_id" { + value = var.external_net +} + +output "router_id" { + value = module.network.router_id +} + +output "k8s_master_fips" { + value = var.number_of_k8s_masters + var.number_of_k8s_masters_no_etcd > 0 ? concat(module.ips.k8s_master_fips, module.ips.k8s_master_no_etcd_fips) : [for key, value in module.ips.k8s_masters_fips : value.address] +} + +output "k8s_node_fips" { + value = var.number_of_k8s_nodes > 0 ? module.ips.k8s_node_fips : [for key, value in module.ips.k8s_nodes_fips : value.address] +} + +output "bastion_fips" { + value = module.ips.bastion_fips +} diff --git a/kubespray/contrib/terraform/openstack/modules/compute/ansible_bastion_template.txt b/kubespray/contrib/terraform/openstack/modules/compute/ansible_bastion_template.txt new file mode 100644 index 0000000..a304b2c --- /dev/null +++ b/kubespray/contrib/terraform/openstack/modules/compute/ansible_bastion_template.txt @@ -0,0 +1 @@ +ansible_ssh_common_args: "-o ProxyCommand='ssh -o StrictHostKeyChecking=no -o UserKnownHostsFile=/dev/null -W %h:%p -q USER@BASTION_ADDRESS {% if ansible_ssh_private_key_file is defined %}-i {{ ansible_ssh_private_key_file }}{% endif %}'" diff --git a/kubespray/contrib/terraform/openstack/modules/compute/main.tf b/kubespray/contrib/terraform/openstack/modules/compute/main.tf new file mode 100644 index 0000000..64ccc7f --- /dev/null +++ b/kubespray/contrib/terraform/openstack/modules/compute/main.tf @@ -0,0 +1,971 @@ +data "openstack_images_image_v2" "vm_image" { + count = var.image_uuid == "" ? 1 : 0 + most_recent = true + name = var.image +} + +data "openstack_images_image_v2" "gfs_image" { + count = var.image_gfs_uuid == "" ? var.image_uuid == "" ? 1 : 0 : 0 + most_recent = true + name = var.image_gfs == "" ? var.image : var.image_gfs +} + +data "openstack_images_image_v2" "image_master" { + count = var.image_master_uuid == "" ? var.image_uuid == "" ? 1 : 0 : 0 + name = var.image_master == "" ? var.image : var.image_master +} + +data "cloudinit_config" "cloudinit" { + part { + content_type = "text/cloud-config" + content = templatefile("${path.module}/templates/cloudinit.yaml.tmpl", { + extra_partitions = [], + netplan_critical_dhcp_interface = "" + }) + } +} + +data "openstack_networking_network_v2" "k8s_network" { + count = var.use_existing_network ? 1 : 0 + name = var.network_name +} + +resource "openstack_compute_keypair_v2" "k8s" { + name = "kubernetes-${var.cluster_name}" + public_key = chomp(file(var.public_key_path)) +} + +resource "openstack_networking_secgroup_v2" "k8s_master" { + name = "${var.cluster_name}-k8s-master" + description = "${var.cluster_name} - Kubernetes Master" + delete_default_rules = true +} + +resource "openstack_networking_secgroup_v2" "k8s_master_extra" { + count = "%{if var.extra_sec_groups}1%{else}0%{endif}" + name = "${var.cluster_name}-k8s-master-${var.extra_sec_groups_name}" + description = "${var.cluster_name} - Kubernetes Master nodes - rules not managed by terraform" + delete_default_rules = true +} + +resource "openstack_networking_secgroup_rule_v2" "k8s_master" { + count = length(var.master_allowed_remote_ips) + direction = "ingress" + ethertype = "IPv4" + protocol = "tcp" + port_range_min = "6443" + port_range_max = "6443" + remote_ip_prefix = var.master_allowed_remote_ips[count.index] + security_group_id = openstack_networking_secgroup_v2.k8s_master.id +} + +resource "openstack_networking_secgroup_rule_v2" "k8s_master_ports" { + count = length(var.master_allowed_ports) + direction = "ingress" + ethertype = "IPv4" + protocol = lookup(var.master_allowed_ports[count.index], "protocol", "tcp") + port_range_min = lookup(var.master_allowed_ports[count.index], "port_range_min") + port_range_max = lookup(var.master_allowed_ports[count.index], "port_range_max") + remote_ip_prefix = lookup(var.master_allowed_ports[count.index], "remote_ip_prefix", "0.0.0.0/0") + security_group_id = openstack_networking_secgroup_v2.k8s_master.id +} + +resource "openstack_networking_secgroup_v2" "bastion" { + name = "${var.cluster_name}-bastion" + count = var.number_of_bastions != "" ? 1 : 0 + description = "${var.cluster_name} - Bastion Server" + delete_default_rules = true +} + +resource "openstack_networking_secgroup_rule_v2" "bastion" { + count = var.number_of_bastions != "" ? length(var.bastion_allowed_remote_ips) : 0 + direction = "ingress" + ethertype = "IPv4" + protocol = "tcp" + port_range_min = "22" + port_range_max = "22" + remote_ip_prefix = var.bastion_allowed_remote_ips[count.index] + security_group_id = openstack_networking_secgroup_v2.bastion[0].id +} + +resource "openstack_networking_secgroup_rule_v2" "k8s_bastion_ports" { + count = length(var.bastion_allowed_ports) + direction = "ingress" + ethertype = "IPv4" + protocol = lookup(var.bastion_allowed_ports[count.index], "protocol", "tcp") + port_range_min = lookup(var.bastion_allowed_ports[count.index], "port_range_min") + port_range_max = lookup(var.bastion_allowed_ports[count.index], "port_range_max") + remote_ip_prefix = lookup(var.bastion_allowed_ports[count.index], "remote_ip_prefix", "0.0.0.0/0") + security_group_id = openstack_networking_secgroup_v2.bastion[0].id +} + +resource "openstack_networking_secgroup_v2" "k8s" { + name = "${var.cluster_name}-k8s" + description = "${var.cluster_name} - Kubernetes" + delete_default_rules = true +} + +resource "openstack_networking_secgroup_rule_v2" "k8s" { + direction = "ingress" + ethertype = "IPv4" + remote_group_id = openstack_networking_secgroup_v2.k8s.id + security_group_id = openstack_networking_secgroup_v2.k8s.id +} + +resource "openstack_networking_secgroup_rule_v2" "k8s_allowed_remote_ips" { + count = length(var.k8s_allowed_remote_ips) + direction = "ingress" + ethertype = "IPv4" + protocol = "tcp" + port_range_min = "22" + port_range_max = "22" + remote_ip_prefix = var.k8s_allowed_remote_ips[count.index] + security_group_id = openstack_networking_secgroup_v2.k8s.id +} + +resource "openstack_networking_secgroup_rule_v2" "egress" { + count = length(var.k8s_allowed_egress_ips) + direction = "egress" + ethertype = "IPv4" + remote_ip_prefix = var.k8s_allowed_egress_ips[count.index] + security_group_id = openstack_networking_secgroup_v2.k8s.id +} + +resource "openstack_networking_secgroup_v2" "worker" { + name = "${var.cluster_name}-k8s-worker" + description = "${var.cluster_name} - Kubernetes worker nodes" + delete_default_rules = true +} + +resource "openstack_networking_secgroup_v2" "worker_extra" { + count = "%{if var.extra_sec_groups}1%{else}0%{endif}" + name = "${var.cluster_name}-k8s-worker-${var.extra_sec_groups_name}" + description = "${var.cluster_name} - Kubernetes worker nodes - rules not managed by terraform" + delete_default_rules = true +} + +resource "openstack_networking_secgroup_rule_v2" "worker" { + count = length(var.worker_allowed_ports) + direction = "ingress" + ethertype = "IPv4" + protocol = lookup(var.worker_allowed_ports[count.index], "protocol", "tcp") + port_range_min = lookup(var.worker_allowed_ports[count.index], "port_range_min") + port_range_max = lookup(var.worker_allowed_ports[count.index], "port_range_max") + remote_ip_prefix = lookup(var.worker_allowed_ports[count.index], "remote_ip_prefix", "0.0.0.0/0") + security_group_id = openstack_networking_secgroup_v2.worker.id +} + +resource "openstack_compute_servergroup_v2" "k8s_master" { + count = var.master_server_group_policy != "" ? 1 : 0 + name = "k8s-master-srvgrp" + policies = [var.master_server_group_policy] +} + +resource "openstack_compute_servergroup_v2" "k8s_node" { + count = var.node_server_group_policy != "" ? 1 : 0 + name = "k8s-node-srvgrp" + policies = [var.node_server_group_policy] +} + +resource "openstack_compute_servergroup_v2" "k8s_etcd" { + count = var.etcd_server_group_policy != "" ? 1 : 0 + name = "k8s-etcd-srvgrp" + policies = [var.etcd_server_group_policy] +} + +resource "openstack_compute_servergroup_v2" "k8s_node_additional" { + for_each = var.additional_server_groups + name = "k8s-${each.key}-srvgrp" + policies = [each.value.policy] +} + +locals { +# master groups + master_sec_groups = compact([ + openstack_networking_secgroup_v2.k8s_master.id, + openstack_networking_secgroup_v2.k8s.id, + var.extra_sec_groups ?openstack_networking_secgroup_v2.k8s_master_extra[0].id : "", + ]) +# worker groups + worker_sec_groups = compact([ + openstack_networking_secgroup_v2.k8s.id, + openstack_networking_secgroup_v2.worker.id, + var.extra_sec_groups ? openstack_networking_secgroup_v2.worker_extra[0].id : "", + ]) +# bastion groups + bastion_sec_groups = compact(concat([ + openstack_networking_secgroup_v2.k8s.id, + openstack_networking_secgroup_v2.bastion[0].id, + ])) +# etcd groups + etcd_sec_groups = compact([openstack_networking_secgroup_v2.k8s.id]) +# glusterfs groups + gfs_sec_groups = compact([openstack_networking_secgroup_v2.k8s.id]) + +# Image uuid + image_to_use_node = var.image_uuid != "" ? var.image_uuid : data.openstack_images_image_v2.vm_image[0].id +# Image_gfs uuid + image_to_use_gfs = var.image_gfs_uuid != "" ? var.image_gfs_uuid : var.image_uuid != "" ? var.image_uuid : data.openstack_images_image_v2.gfs_image[0].id +# image_master uuidimage_gfs_uuid + image_to_use_master = var.image_master_uuid != "" ? var.image_master_uuid : var.image_uuid != "" ? var.image_uuid : data.openstack_images_image_v2.image_master[0].id + + k8s_nodes_settings = { + for name, node in var.k8s_nodes : + name => { + "use_local_disk" = (node.root_volume_size_in_gb != null ? node.root_volume_size_in_gb : var.node_root_volume_size_in_gb) == 0, + "image_id" = node.image_id != null ? node.image_id : local.image_to_use_node, + "volume_size" = node.root_volume_size_in_gb != null ? node.root_volume_size_in_gb : var.node_root_volume_size_in_gb, + "volume_type" = node.volume_type != null ? node.volume_type : var.node_volume_type, + "network_id" = node.network_id != null ? node.network_id : (var.use_existing_network ? data.openstack_networking_network_v2.k8s_network[0].id : var.network_id) + "server_group" = node.server_group != null ? [openstack_compute_servergroup_v2.k8s_node_additional[node.server_group].id] : (var.node_server_group_policy != "" ? [openstack_compute_servergroup_v2.k8s_node[0].id] : []) + } + } + + k8s_masters_settings = { + for name, node in var.k8s_masters : + name => { + "use_local_disk" = (node.root_volume_size_in_gb != null ? node.root_volume_size_in_gb : var.master_root_volume_size_in_gb) == 0, + "image_id" = node.image_id != null ? node.image_id : local.image_to_use_master, + "volume_size" = node.root_volume_size_in_gb != null ? node.root_volume_size_in_gb : var.master_root_volume_size_in_gb, + "volume_type" = node.volume_type != null ? node.volume_type : var.master_volume_type, + "network_id" = node.network_id != null ? node.network_id : (var.use_existing_network ? data.openstack_networking_network_v2.k8s_network[0].id : var.network_id) + } + } +} + +resource "openstack_networking_port_v2" "bastion_port" { + count = var.number_of_bastions + name = "${var.cluster_name}-bastion-${count.index + 1}" + network_id = var.use_existing_network ? data.openstack_networking_network_v2.k8s_network[0].id : var.network_id + admin_state_up = "true" + port_security_enabled = var.force_null_port_security ? null : var.port_security_enabled + security_group_ids = var.port_security_enabled ? local.bastion_sec_groups : null + no_security_groups = var.port_security_enabled ? null : false + dynamic "fixed_ip" { + for_each = var.private_subnet_id == "" ? [] : [true] + content { + subnet_id = var.private_subnet_id + } + } + + depends_on = [ + var.network_router_id + ] +} + +resource "openstack_compute_instance_v2" "bastion" { + name = "${var.cluster_name}-bastion-${count.index + 1}" + count = var.number_of_bastions + image_id = var.bastion_root_volume_size_in_gb == 0 ? local.image_to_use_node : null + flavor_id = var.flavor_bastion + key_pair = openstack_compute_keypair_v2.k8s.name + user_data = data.cloudinit_config.cloudinit.rendered + + dynamic "block_device" { + for_each = var.bastion_root_volume_size_in_gb > 0 ? [local.image_to_use_node] : [] + content { + uuid = local.image_to_use_node + source_type = "image" + volume_size = var.bastion_root_volume_size_in_gb + boot_index = 0 + destination_type = "volume" + delete_on_termination = true + } + } + + network { + port = element(openstack_networking_port_v2.bastion_port.*.id, count.index) + } + + metadata = { + ssh_user = var.ssh_user + kubespray_groups = "bastion" + depends_on = var.network_router_id + use_access_ip = var.use_access_ip + } + + provisioner "local-exec" { + command = "sed -e s/USER/${var.ssh_user}/ -e s/BASTION_ADDRESS/${var.bastion_fips[0]}/ ${path.module}/ansible_bastion_template.txt > ${var.group_vars_path}/no_floating.yml" + } +} + +resource "openstack_networking_port_v2" "k8s_master_port" { + count = var.number_of_k8s_masters + name = "${var.cluster_name}-k8s-master-${count.index + 1}" + network_id = var.use_existing_network ? data.openstack_networking_network_v2.k8s_network[0].id : var.network_id + admin_state_up = "true" + port_security_enabled = var.force_null_port_security ? null : var.port_security_enabled + security_group_ids = var.port_security_enabled ? local.master_sec_groups : null + no_security_groups = var.port_security_enabled ? null : false + dynamic "fixed_ip" { + for_each = var.private_subnet_id == "" ? [] : [true] + content { + subnet_id = var.private_subnet_id + } + } + + depends_on = [ + var.network_router_id + ] +} + +resource "openstack_compute_instance_v2" "k8s_master" { + name = "${var.cluster_name}-k8s-master-${count.index + 1}" + count = var.number_of_k8s_masters + availability_zone = element(var.az_list, count.index) + image_id = var.master_root_volume_size_in_gb == 0 ? local.image_to_use_master : null + flavor_id = var.flavor_k8s_master + key_pair = openstack_compute_keypair_v2.k8s.name + user_data = data.cloudinit_config.cloudinit.rendered + + + dynamic "block_device" { + for_each = var.master_root_volume_size_in_gb > 0 ? [local.image_to_use_master] : [] + content { + uuid = local.image_to_use_master + source_type = "image" + volume_size = var.master_root_volume_size_in_gb + volume_type = var.master_volume_type + boot_index = 0 + destination_type = "volume" + delete_on_termination = true + } + } + + network { + port = element(openstack_networking_port_v2.k8s_master_port.*.id, count.index) + } + + dynamic "scheduler_hints" { + for_each = var.master_server_group_policy != "" ? [openstack_compute_servergroup_v2.k8s_master[0]] : [] + content { + group = openstack_compute_servergroup_v2.k8s_master[0].id + } + } + + metadata = { + ssh_user = var.ssh_user + kubespray_groups = "etcd,kube_control_plane,${var.supplementary_master_groups},k8s_cluster" + depends_on = var.network_router_id + use_access_ip = var.use_access_ip + } + + provisioner "local-exec" { + command = "sed -e s/USER/${var.ssh_user}/ -e s/BASTION_ADDRESS/${element(concat(var.bastion_fips, var.k8s_master_fips), 0)}/ ${path.module}/ansible_bastion_template.txt > ${var.group_vars_path}/no_floating.yml" + } +} + +resource "openstack_networking_port_v2" "k8s_masters_port" { + for_each = var.number_of_k8s_masters == 0 && var.number_of_k8s_masters_no_etcd == 0 && var.number_of_k8s_masters_no_floating_ip == 0 && var.number_of_k8s_masters_no_floating_ip_no_etcd == 0 ? var.k8s_masters : {} + name = "${var.cluster_name}-k8s-${each.key}" + network_id = local.k8s_masters_settings[each.key].network_id + admin_state_up = "true" + port_security_enabled = var.force_null_port_security ? null : var.port_security_enabled + security_group_ids = var.port_security_enabled ? local.master_sec_groups : null + no_security_groups = var.port_security_enabled ? null : false + dynamic "fixed_ip" { + for_each = var.private_subnet_id == "" ? [] : [true] + content { + subnet_id = var.private_subnet_id + } + } + + depends_on = [ + var.network_router_id + ] +} + +resource "openstack_compute_instance_v2" "k8s_masters" { + for_each = var.number_of_k8s_masters == 0 && var.number_of_k8s_masters_no_etcd == 0 && var.number_of_k8s_masters_no_floating_ip == 0 && var.number_of_k8s_masters_no_floating_ip_no_etcd == 0 ? var.k8s_masters : {} + name = "${var.cluster_name}-k8s-${each.key}" + availability_zone = each.value.az + image_id = local.k8s_masters_settings[each.key].use_local_disk ? local.k8s_masters_settings[each.key].image_id : null + flavor_id = each.value.flavor + key_pair = openstack_compute_keypair_v2.k8s.name + + dynamic "block_device" { + for_each = !local.k8s_masters_settings[each.key].use_local_disk ? [local.k8s_masters_settings[each.key].image_id] : [] + content { + uuid = block_device.value + source_type = "image" + volume_size = local.k8s_masters_settings[each.key].volume_size + volume_type = local.k8s_masters_settings[each.key].volume_type + boot_index = 0 + destination_type = "volume" + delete_on_termination = true + } + } + + network { + port = openstack_networking_port_v2.k8s_masters_port[each.key].id + } + + dynamic "scheduler_hints" { + for_each = var.master_server_group_policy != "" ? [openstack_compute_servergroup_v2.k8s_master[0]] : [] + content { + group = openstack_compute_servergroup_v2.k8s_master[0].id + } + } + + metadata = { + ssh_user = var.ssh_user + kubespray_groups = "%{if each.value.etcd == true}etcd,%{endif}kube_control_plane,${var.supplementary_master_groups},k8s_cluster%{if each.value.floating_ip == false},no_floating%{endif}" + depends_on = var.network_router_id + use_access_ip = var.use_access_ip + } + + provisioner "local-exec" { + command = "%{if each.value.floating_ip}sed s/USER/${var.ssh_user}/ ${path.module}/ansible_bastion_template.txt | sed s/BASTION_ADDRESS/${element(concat(var.bastion_fips, [for key, value in var.k8s_masters_fips : value.address]), 0)}/ > ${var.group_vars_path}/no_floating.yml%{else}true%{endif}" + } +} + +resource "openstack_networking_port_v2" "k8s_master_no_etcd_port" { + count = var.number_of_k8s_masters_no_etcd + name = "${var.cluster_name}-k8s-master-ne-${count.index + 1}" + network_id = var.use_existing_network ? data.openstack_networking_network_v2.k8s_network[0].id : var.network_id + admin_state_up = "true" + port_security_enabled = var.force_null_port_security ? null : var.port_security_enabled + security_group_ids = var.port_security_enabled ? local.master_sec_groups : null + no_security_groups = var.port_security_enabled ? null : false + dynamic "fixed_ip" { + for_each = var.private_subnet_id == "" ? [] : [true] + content { + subnet_id = var.private_subnet_id + } + } + + depends_on = [ + var.network_router_id + ] +} + +resource "openstack_compute_instance_v2" "k8s_master_no_etcd" { + name = "${var.cluster_name}-k8s-master-ne-${count.index + 1}" + count = var.number_of_k8s_masters_no_etcd + availability_zone = element(var.az_list, count.index) + image_id = var.master_root_volume_size_in_gb == 0 ? local.image_to_use_master : null + flavor_id = var.flavor_k8s_master + key_pair = openstack_compute_keypair_v2.k8s.name + user_data = data.cloudinit_config.cloudinit.rendered + + + dynamic "block_device" { + for_each = var.master_root_volume_size_in_gb > 0 ? [local.image_to_use_master] : [] + content { + uuid = local.image_to_use_master + source_type = "image" + volume_size = var.master_root_volume_size_in_gb + volume_type = var.master_volume_type + boot_index = 0 + destination_type = "volume" + delete_on_termination = true + } + } + + network { + port = element(openstack_networking_port_v2.k8s_master_no_etcd_port.*.id, count.index) + } + + dynamic "scheduler_hints" { + for_each = var.master_server_group_policy != "" ? [openstack_compute_servergroup_v2.k8s_master[0]] : [] + content { + group = openstack_compute_servergroup_v2.k8s_master[0].id + } + } + + metadata = { + ssh_user = var.ssh_user + kubespray_groups = "kube_control_plane,${var.supplementary_master_groups},k8s_cluster" + depends_on = var.network_router_id + use_access_ip = var.use_access_ip + } + + provisioner "local-exec" { + command = "sed -e s/USER/${var.ssh_user}/ -e s/BASTION_ADDRESS/${element(concat(var.bastion_fips, var.k8s_master_fips), 0)}/ ${path.module}/ansible_bastion_template.txt > ${var.group_vars_path}/no_floating.yml" + } +} + +resource "openstack_networking_port_v2" "etcd_port" { + count = var.number_of_etcd + name = "${var.cluster_name}-etcd-${count.index + 1}" + network_id = var.use_existing_network ? data.openstack_networking_network_v2.k8s_network[0].id : var.network_id + admin_state_up = "true" + port_security_enabled = var.force_null_port_security ? null : var.port_security_enabled + security_group_ids = var.port_security_enabled ? local.etcd_sec_groups : null + no_security_groups = var.port_security_enabled ? null : false + dynamic "fixed_ip" { + for_each = var.private_subnet_id == "" ? [] : [true] + content { + subnet_id = var.private_subnet_id + } + } + + depends_on = [ + var.network_router_id + ] +} + +resource "openstack_compute_instance_v2" "etcd" { + name = "${var.cluster_name}-etcd-${count.index + 1}" + count = var.number_of_etcd + availability_zone = element(var.az_list, count.index) + image_id = var.etcd_root_volume_size_in_gb == 0 ? local.image_to_use_master : null + flavor_id = var.flavor_etcd + key_pair = openstack_compute_keypair_v2.k8s.name + user_data = data.cloudinit_config.cloudinit.rendered + + dynamic "block_device" { + for_each = var.etcd_root_volume_size_in_gb > 0 ? [local.image_to_use_master] : [] + content { + uuid = local.image_to_use_master + source_type = "image" + volume_size = var.etcd_root_volume_size_in_gb + boot_index = 0 + destination_type = "volume" + delete_on_termination = true + } + } + + network { + port = element(openstack_networking_port_v2.etcd_port.*.id, count.index) + } + + dynamic "scheduler_hints" { + for_each = var.etcd_server_group_policy != "" ? [openstack_compute_servergroup_v2.k8s_etcd[0]] : [] + content { + group = openstack_compute_servergroup_v2.k8s_etcd[0].id + } + } + + metadata = { + ssh_user = var.ssh_user + kubespray_groups = "etcd,no_floating" + depends_on = var.network_router_id + use_access_ip = var.use_access_ip + } +} + +resource "openstack_networking_port_v2" "k8s_master_no_floating_ip_port" { + count = var.number_of_k8s_masters_no_floating_ip + name = "${var.cluster_name}-k8s-master-nf-${count.index + 1}" + network_id = var.use_existing_network ? data.openstack_networking_network_v2.k8s_network[0].id : var.network_id + admin_state_up = "true" + port_security_enabled = var.force_null_port_security ? null : var.port_security_enabled + security_group_ids = var.port_security_enabled ? local.master_sec_groups : null + no_security_groups = var.port_security_enabled ? null : false + dynamic "fixed_ip" { + for_each = var.private_subnet_id == "" ? [] : [true] + content { + subnet_id = var.private_subnet_id + } + } + + depends_on = [ + var.network_router_id + ] +} + +resource "openstack_compute_instance_v2" "k8s_master_no_floating_ip" { + name = "${var.cluster_name}-k8s-master-nf-${count.index + 1}" + count = var.number_of_k8s_masters_no_floating_ip + availability_zone = element(var.az_list, count.index) + image_id = var.master_root_volume_size_in_gb == 0 ? local.image_to_use_master : null + flavor_id = var.flavor_k8s_master + key_pair = openstack_compute_keypair_v2.k8s.name + + dynamic "block_device" { + for_each = var.master_root_volume_size_in_gb > 0 ? [local.image_to_use_master] : [] + content { + uuid = local.image_to_use_master + source_type = "image" + volume_size = var.master_root_volume_size_in_gb + volume_type = var.master_volume_type + boot_index = 0 + destination_type = "volume" + delete_on_termination = true + } + } + + network { + port = element(openstack_networking_port_v2.k8s_master_no_floating_ip_port.*.id, count.index) + } + + dynamic "scheduler_hints" { + for_each = var.master_server_group_policy != "" ? [openstack_compute_servergroup_v2.k8s_master[0]] : [] + content { + group = openstack_compute_servergroup_v2.k8s_master[0].id + } + } + + metadata = { + ssh_user = var.ssh_user + kubespray_groups = "etcd,kube_control_plane,${var.supplementary_master_groups},k8s_cluster,no_floating" + depends_on = var.network_router_id + use_access_ip = var.use_access_ip + } +} + +resource "openstack_networking_port_v2" "k8s_master_no_floating_ip_no_etcd_port" { + count = var.number_of_k8s_masters_no_floating_ip_no_etcd + name = "${var.cluster_name}-k8s-master-ne-nf-${count.index + 1}" + network_id = var.use_existing_network ? data.openstack_networking_network_v2.k8s_network[0].id : var.network_id + admin_state_up = "true" + port_security_enabled = var.force_null_port_security ? null : var.port_security_enabled + security_group_ids = var.port_security_enabled ? local.master_sec_groups : null + no_security_groups = var.port_security_enabled ? null : false + dynamic "fixed_ip" { + for_each = var.private_subnet_id == "" ? [] : [true] + content { + subnet_id = var.private_subnet_id + } + } + + depends_on = [ + var.network_router_id + ] +} + +resource "openstack_compute_instance_v2" "k8s_master_no_floating_ip_no_etcd" { + name = "${var.cluster_name}-k8s-master-ne-nf-${count.index + 1}" + count = var.number_of_k8s_masters_no_floating_ip_no_etcd + availability_zone = element(var.az_list, count.index) + image_id = var.master_root_volume_size_in_gb == 0 ? local.image_to_use_master : null + flavor_id = var.flavor_k8s_master + key_pair = openstack_compute_keypair_v2.k8s.name + user_data = data.cloudinit_config.cloudinit.rendered + + dynamic "block_device" { + for_each = var.master_root_volume_size_in_gb > 0 ? [local.image_to_use_master] : [] + content { + uuid = local.image_to_use_master + source_type = "image" + volume_size = var.master_root_volume_size_in_gb + volume_type = var.master_volume_type + boot_index = 0 + destination_type = "volume" + delete_on_termination = true + } + } + + network { + port = element(openstack_networking_port_v2.k8s_master_no_floating_ip_no_etcd_port.*.id, count.index) + } + + dynamic "scheduler_hints" { + for_each = var.master_server_group_policy != "" ? [openstack_compute_servergroup_v2.k8s_master[0]] : [] + content { + group = openstack_compute_servergroup_v2.k8s_master[0].id + } + } + + metadata = { + ssh_user = var.ssh_user + kubespray_groups = "kube_control_plane,${var.supplementary_master_groups},k8s_cluster,no_floating" + depends_on = var.network_router_id + use_access_ip = var.use_access_ip + } +} + +resource "openstack_networking_port_v2" "k8s_node_port" { + count = var.number_of_k8s_nodes + name = "${var.cluster_name}-k8s-node-${count.index + 1}" + network_id = var.use_existing_network ? data.openstack_networking_network_v2.k8s_network[0].id : var.network_id + admin_state_up = "true" + port_security_enabled = var.force_null_port_security ? null : var.port_security_enabled + security_group_ids = var.port_security_enabled ? local.worker_sec_groups : null + no_security_groups = var.port_security_enabled ? null : false + dynamic "fixed_ip" { + for_each = var.private_subnet_id == "" ? [] : [true] + content { + subnet_id = var.private_subnet_id + } + } + + depends_on = [ + var.network_router_id + ] +} + +resource "openstack_compute_instance_v2" "k8s_node" { + name = "${var.cluster_name}-k8s-node-${count.index + 1}" + count = var.number_of_k8s_nodes + availability_zone = element(var.az_list_node, count.index) + image_id = var.node_root_volume_size_in_gb == 0 ? local.image_to_use_node : null + flavor_id = var.flavor_k8s_node + key_pair = openstack_compute_keypair_v2.k8s.name + user_data = data.cloudinit_config.cloudinit.rendered + + dynamic "block_device" { + for_each = var.node_root_volume_size_in_gb > 0 ? [local.image_to_use_node] : [] + content { + uuid = local.image_to_use_node + source_type = "image" + volume_size = var.node_root_volume_size_in_gb + volume_type = var.node_volume_type + boot_index = 0 + destination_type = "volume" + delete_on_termination = true + } + } + + network { + port = element(openstack_networking_port_v2.k8s_node_port.*.id, count.index) + } + + + dynamic "scheduler_hints" { + for_each = var.node_server_group_policy != "" ? [openstack_compute_servergroup_v2.k8s_node[0]] : [] + content { + group = openstack_compute_servergroup_v2.k8s_node[0].id + } + } + + metadata = { + ssh_user = var.ssh_user + kubespray_groups = "kube_node,k8s_cluster,${var.supplementary_node_groups}" + depends_on = var.network_router_id + use_access_ip = var.use_access_ip + } + + provisioner "local-exec" { + command = "sed -e s/USER/${var.ssh_user}/ -e s/BASTION_ADDRESS/${element(concat(var.bastion_fips, var.k8s_node_fips), 0)}/ ${path.module}/ansible_bastion_template.txt > ${var.group_vars_path}/no_floating.yml" + } +} + +resource "openstack_networking_port_v2" "k8s_node_no_floating_ip_port" { + count = var.number_of_k8s_nodes_no_floating_ip + name = "${var.cluster_name}-k8s-node-nf-${count.index + 1}" + network_id = var.use_existing_network ? data.openstack_networking_network_v2.k8s_network[0].id : var.network_id + admin_state_up = "true" + port_security_enabled = var.force_null_port_security ? null : var.port_security_enabled + security_group_ids = var.port_security_enabled ? local.worker_sec_groups : null + no_security_groups = var.port_security_enabled ? null : false + dynamic "fixed_ip" { + for_each = var.private_subnet_id == "" ? [] : [true] + content { + subnet_id = var.private_subnet_id + } + } + + depends_on = [ + var.network_router_id + ] +} + +resource "openstack_compute_instance_v2" "k8s_node_no_floating_ip" { + name = "${var.cluster_name}-k8s-node-nf-${count.index + 1}" + count = var.number_of_k8s_nodes_no_floating_ip + availability_zone = element(var.az_list_node, count.index) + image_id = var.node_root_volume_size_in_gb == 0 ? local.image_to_use_node : null + flavor_id = var.flavor_k8s_node + key_pair = openstack_compute_keypair_v2.k8s.name + user_data = data.cloudinit_config.cloudinit.rendered + + dynamic "block_device" { + for_each = var.node_root_volume_size_in_gb > 0 ? [local.image_to_use_node] : [] + content { + uuid = local.image_to_use_node + source_type = "image" + volume_size = var.node_root_volume_size_in_gb + volume_type = var.node_volume_type + boot_index = 0 + destination_type = "volume" + delete_on_termination = true + } + } + + network { + port = element(openstack_networking_port_v2.k8s_node_no_floating_ip_port.*.id, count.index) + } + + dynamic "scheduler_hints" { + for_each = var.node_server_group_policy != "" ? [openstack_compute_servergroup_v2.k8s_node[0].id] : [] + content { + group = scheduler_hints.value + } + } + + metadata = { + ssh_user = var.ssh_user + kubespray_groups = "kube_node,k8s_cluster,no_floating,${var.supplementary_node_groups}" + depends_on = var.network_router_id + use_access_ip = var.use_access_ip + } +} + +resource "openstack_networking_port_v2" "k8s_nodes_port" { + for_each = var.number_of_k8s_nodes == 0 && var.number_of_k8s_nodes_no_floating_ip == 0 ? var.k8s_nodes : {} + name = "${var.cluster_name}-k8s-node-${each.key}" + network_id = local.k8s_nodes_settings[each.key].network_id + admin_state_up = "true" + port_security_enabled = var.force_null_port_security ? null : var.port_security_enabled + security_group_ids = var.port_security_enabled ? local.worker_sec_groups : null + no_security_groups = var.port_security_enabled ? null : false + dynamic "fixed_ip" { + for_each = var.private_subnet_id == "" ? [] : [true] + content { + subnet_id = var.private_subnet_id + } + } + + depends_on = [ + var.network_router_id + ] +} + +resource "openstack_compute_instance_v2" "k8s_nodes" { + for_each = var.number_of_k8s_nodes == 0 && var.number_of_k8s_nodes_no_floating_ip == 0 ? var.k8s_nodes : {} + name = "${var.cluster_name}-k8s-node-${each.key}" + availability_zone = each.value.az + image_id = local.k8s_nodes_settings[each.key].use_local_disk ? local.k8s_nodes_settings[each.key].image_id : null + flavor_id = each.value.flavor + key_pair = openstack_compute_keypair_v2.k8s.name + user_data = each.value.cloudinit != null ? templatefile("${path.module}/templates/cloudinit.yaml.tmpl", { + extra_partitions = each.value.cloudinit.extra_partitions, + netplan_critical_dhcp_interface = each.value.cloudinit.netplan_critical_dhcp_interface, + }) : data.cloudinit_config.cloudinit.rendered + + dynamic "block_device" { + for_each = !local.k8s_nodes_settings[each.key].use_local_disk ? [local.k8s_nodes_settings[each.key].image_id] : [] + content { + uuid = block_device.value + source_type = "image" + volume_size = local.k8s_nodes_settings[each.key].volume_size + volume_type = local.k8s_nodes_settings[each.key].volume_type + boot_index = 0 + destination_type = "volume" + delete_on_termination = true + } + } + + network { + port = openstack_networking_port_v2.k8s_nodes_port[each.key].id + } + + dynamic "scheduler_hints" { + for_each = local.k8s_nodes_settings[each.key].server_group + content { + group = scheduler_hints.value + } + } + + metadata = { + ssh_user = var.ssh_user + kubespray_groups = "kube_node,k8s_cluster,%{if each.value.floating_ip == false}no_floating,%{endif}${var.supplementary_node_groups}${each.value.extra_groups != null ? ",${each.value.extra_groups}" : ""}" + depends_on = var.network_router_id + use_access_ip = var.use_access_ip + } + + provisioner "local-exec" { + command = "%{if each.value.floating_ip}sed -e s/USER/${var.ssh_user}/ -e s/BASTION_ADDRESS/${element(concat(var.bastion_fips, [for key, value in var.k8s_nodes_fips : value.address]), 0)}/ ${path.module}/ansible_bastion_template.txt > ${var.group_vars_path}/no_floating.yml%{else}true%{endif}" + } +} + +resource "openstack_networking_port_v2" "glusterfs_node_no_floating_ip_port" { + count = var.number_of_gfs_nodes_no_floating_ip + name = "${var.cluster_name}-gfs-node-nf-${count.index + 1}" + network_id = var.use_existing_network ? data.openstack_networking_network_v2.k8s_network[0].id : var.network_id + admin_state_up = "true" + port_security_enabled = var.force_null_port_security ? null : var.port_security_enabled + security_group_ids = var.port_security_enabled ? local.gfs_sec_groups : null + no_security_groups = var.port_security_enabled ? null : false + dynamic "fixed_ip" { + for_each = var.private_subnet_id == "" ? [] : [true] + content { + subnet_id = var.private_subnet_id + } + } + + depends_on = [ + var.network_router_id + ] +} + +resource "openstack_compute_instance_v2" "glusterfs_node_no_floating_ip" { + name = "${var.cluster_name}-gfs-node-nf-${count.index + 1}" + count = var.number_of_gfs_nodes_no_floating_ip + availability_zone = element(var.az_list, count.index) + image_name = var.gfs_root_volume_size_in_gb == 0 ? local.image_to_use_gfs : null + flavor_id = var.flavor_gfs_node + key_pair = openstack_compute_keypair_v2.k8s.name + + dynamic "block_device" { + for_each = var.gfs_root_volume_size_in_gb > 0 ? [local.image_to_use_gfs] : [] + content { + uuid = local.image_to_use_gfs + source_type = "image" + volume_size = var.gfs_root_volume_size_in_gb + boot_index = 0 + destination_type = "volume" + delete_on_termination = true + } + } + + network { + port = element(openstack_networking_port_v2.glusterfs_node_no_floating_ip_port.*.id, count.index) + } + + dynamic "scheduler_hints" { + for_each = var.node_server_group_policy != "" ? [openstack_compute_servergroup_v2.k8s_node[0]] : [] + content { + group = openstack_compute_servergroup_v2.k8s_node[0].id + } + } + + metadata = { + ssh_user = var.ssh_user_gfs + kubespray_groups = "gfs-cluster,network-storage,no_floating" + depends_on = var.network_router_id + use_access_ip = var.use_access_ip + } +} + +resource "openstack_networking_floatingip_associate_v2" "bastion" { + count = var.number_of_bastions + floating_ip = var.bastion_fips[count.index] + port_id = element(openstack_networking_port_v2.bastion_port.*.id, count.index) +} + + +resource "openstack_networking_floatingip_associate_v2" "k8s_master" { + count = var.number_of_k8s_masters + floating_ip = var.k8s_master_fips[count.index] + port_id = element(openstack_networking_port_v2.k8s_master_port.*.id, count.index) +} + +resource "openstack_networking_floatingip_associate_v2" "k8s_masters" { + for_each = var.number_of_k8s_masters == 0 && var.number_of_k8s_masters_no_etcd == 0 && var.number_of_k8s_masters_no_floating_ip == 0 && var.number_of_k8s_masters_no_floating_ip_no_etcd == 0 ? { for key, value in var.k8s_masters : key => value if value.floating_ip } : {} + floating_ip = var.k8s_masters_fips[each.key].address + port_id = openstack_networking_port_v2.k8s_masters_port[each.key].id +} + +resource "openstack_networking_floatingip_associate_v2" "k8s_master_no_etcd" { + count = var.master_root_volume_size_in_gb == 0 ? var.number_of_k8s_masters_no_etcd : 0 + floating_ip = var.k8s_master_no_etcd_fips[count.index] + port_id = element(openstack_networking_port_v2.k8s_master_no_etcd_port.*.id, count.index) +} + +resource "openstack_networking_floatingip_associate_v2" "k8s_node" { + count = var.node_root_volume_size_in_gb == 0 ? var.number_of_k8s_nodes : 0 + floating_ip = var.k8s_node_fips[count.index] + port_id = element(openstack_networking_port_v2.k8s_node_port.*.id, count.index) +} + +resource "openstack_networking_floatingip_associate_v2" "k8s_nodes" { + for_each = var.number_of_k8s_nodes == 0 && var.number_of_k8s_nodes_no_floating_ip == 0 ? { for key, value in var.k8s_nodes : key => value if value.floating_ip } : {} + floating_ip = var.k8s_nodes_fips[each.key].address + port_id = openstack_networking_port_v2.k8s_nodes_port[each.key].id +} + +resource "openstack_blockstorage_volume_v2" "glusterfs_volume" { + name = "${var.cluster_name}-glusterfs_volume-${count.index + 1}" + count = var.gfs_root_volume_size_in_gb == 0 ? var.number_of_gfs_nodes_no_floating_ip : 0 + description = "Non-ephemeral volume for GlusterFS" + size = var.gfs_volume_size_in_gb +} + +resource "openstack_compute_volume_attach_v2" "glusterfs_volume" { + count = var.gfs_root_volume_size_in_gb == 0 ? var.number_of_gfs_nodes_no_floating_ip : 0 + instance_id = element(openstack_compute_instance_v2.glusterfs_node_no_floating_ip.*.id, count.index) + volume_id = element(openstack_blockstorage_volume_v2.glusterfs_volume.*.id, count.index) +} diff --git a/kubespray/contrib/terraform/openstack/modules/compute/templates/cloudinit.yaml.tmpl b/kubespray/contrib/terraform/openstack/modules/compute/templates/cloudinit.yaml.tmpl new file mode 100644 index 0000000..fd05cc4 --- /dev/null +++ b/kubespray/contrib/terraform/openstack/modules/compute/templates/cloudinit.yaml.tmpl @@ -0,0 +1,54 @@ +%{~ if length(extra_partitions) > 0 || netplan_critical_dhcp_interface != "" } +#cloud-config +bootcmd: +%{~ for idx, partition in extra_partitions } +- [ cloud-init-per, once, move-second-header, sgdisk, --move-second-header, ${partition.volume_path} ] +- [ cloud-init-per, once, create-part-${idx}, parted, --script, ${partition.volume_path}, 'mkpart extended ext4 ${partition.partition_start} ${partition.partition_end}' ] +- [ cloud-init-per, once, create-fs-part-${idx}, mkfs.ext4, ${partition.partition_path} ] +%{~ endfor } + +runcmd: +%{~ if netplan_critical_dhcp_interface != "" } + - netplan apply +%{~ endif } +%{~ for idx, partition in extra_partitions } + - mkdir -p ${partition.mount_path} + - chown nobody:nogroup ${partition.mount_path} + - mount ${partition.partition_path} ${partition.mount_path} +%{~ endfor ~} + +%{~ if netplan_critical_dhcp_interface != "" } +write_files: + - path: /etc/netplan/90-critical-dhcp.yaml + content: | + network: + version: 2 + ethernets: + ${ netplan_critical_dhcp_interface }: + dhcp4: true + critical: true +%{~ endif } + +mounts: +%{~ for idx, partition in extra_partitions } + - [ ${partition.partition_path}, ${partition.mount_path} ] +%{~ endfor } +%{~ else ~} +# yamllint disable rule:comments +#cloud-config +## in some cases novnc console access is required +## it requires ssh password to be set +#ssh_pwauth: yes +#chpasswd: +# list: | +# root:secret +# expire: False + +## in some cases direct root ssh access via ssh key is required +#disable_root: false + +## in some cases additional CA certs are required +#ca-certs: +# trusted: | +# -----BEGIN CERTIFICATE----- +%{~ endif } diff --git a/kubespray/contrib/terraform/openstack/modules/compute/variables.tf b/kubespray/contrib/terraform/openstack/modules/compute/variables.tf new file mode 100644 index 0000000..1a78f50 --- /dev/null +++ b/kubespray/contrib/terraform/openstack/modules/compute/variables.tf @@ -0,0 +1,235 @@ +variable "cluster_name" {} + +variable "az_list" { + type = list(string) +} + +variable "az_list_node" { + type = list(string) +} + +variable "number_of_k8s_masters" {} + +variable "number_of_k8s_masters_no_etcd" {} + +variable "number_of_etcd" {} + +variable "number_of_k8s_masters_no_floating_ip" {} + +variable "number_of_k8s_masters_no_floating_ip_no_etcd" {} + +variable "number_of_k8s_nodes" {} + +variable "number_of_k8s_nodes_no_floating_ip" {} + +variable "number_of_bastions" {} + +variable "number_of_gfs_nodes_no_floating_ip" {} + +variable "bastion_root_volume_size_in_gb" {} + +variable "etcd_root_volume_size_in_gb" {} + +variable "master_root_volume_size_in_gb" {} + +variable "node_root_volume_size_in_gb" {} + +variable "gfs_root_volume_size_in_gb" {} + +variable "gfs_volume_size_in_gb" {} + +variable "master_volume_type" {} + +variable "node_volume_type" {} + +variable "public_key_path" {} + +variable "image" {} + +variable "image_gfs" {} + +variable "ssh_user" {} + +variable "ssh_user_gfs" {} + +variable "flavor_k8s_master" {} + +variable "flavor_k8s_node" {} + +variable "flavor_etcd" {} + +variable "flavor_gfs_node" {} + +variable "network_name" {} + +variable "flavor_bastion" {} + +variable "network_id" { + default = "" +} + +variable "use_existing_network" { + type = bool +} + +variable "network_router_id" { + default = "" +} + +variable "k8s_master_fips" { + type = list +} + +variable "k8s_master_no_etcd_fips" { + type = list +} + +variable "k8s_node_fips" { + type = list +} + +variable "k8s_masters_fips" { + type = map +} + +variable "k8s_nodes_fips" { + type = map +} + +variable "bastion_fips" { + type = list +} + +variable "bastion_allowed_remote_ips" { + type = list +} + +variable "master_allowed_remote_ips" { + type = list +} + +variable "k8s_allowed_remote_ips" { + type = list +} + +variable "k8s_allowed_egress_ips" { + type = list +} + +variable "k8s_masters" { + type = map(object({ + az = string + flavor = string + floating_ip = bool + etcd = bool + image_id = optional(string) + root_volume_size_in_gb = optional(number) + volume_type = optional(string) + network_id = optional(string) + })) +} + +variable "k8s_nodes" { + type = map(object({ + az = string + flavor = string + floating_ip = bool + extra_groups = optional(string) + image_id = optional(string) + root_volume_size_in_gb = optional(number) + volume_type = optional(string) + network_id = optional(string) + additional_server_groups = optional(list(string)) + server_group = optional(string) + cloudinit = optional(object({ + extra_partitions = optional(list(object({ + volume_path = string + partition_path = string + partition_start = string + partition_end = string + mount_path = string + })), []) + netplan_critical_dhcp_interface = optional(string, "") + })) + })) +} + +variable "additional_server_groups" { + type = map(object({ + policy = string + })) +} + +variable "supplementary_master_groups" { + default = "" +} + +variable "supplementary_node_groups" { + default = "" +} + +variable "master_allowed_ports" { + type = list +} + +variable "worker_allowed_ports" { + type = list +} + +variable "bastion_allowed_ports" { + type = list +} + +variable "use_access_ip" {} + +variable "master_server_group_policy" { + type = string +} + +variable "node_server_group_policy" { + type = string +} + +variable "etcd_server_group_policy" { + type = string +} + +variable "extra_sec_groups" { + type = bool +} + +variable "extra_sec_groups_name" { + type = string +} + +variable "image_uuid" { + type = string +} + +variable "image_gfs_uuid" { + type = string +} + +variable "image_master" { + type = string +} + +variable "image_master_uuid" { + type = string +} + +variable "group_vars_path" { + type = string +} + +variable "port_security_enabled" { + type = bool +} + +variable "force_null_port_security" { + type = bool +} + +variable "private_subnet_id" { + type = string +} diff --git a/kubespray/contrib/terraform/openstack/modules/compute/versions.tf b/kubespray/contrib/terraform/openstack/modules/compute/versions.tf new file mode 100644 index 0000000..bfcf77a --- /dev/null +++ b/kubespray/contrib/terraform/openstack/modules/compute/versions.tf @@ -0,0 +1,8 @@ +terraform { + required_providers { + openstack = { + source = "terraform-provider-openstack/openstack" + } + } + required_version = ">= 1.3.0" +} diff --git a/kubespray/contrib/terraform/openstack/modules/ips/main.tf b/kubespray/contrib/terraform/openstack/modules/ips/main.tf new file mode 100644 index 0000000..b0989dc --- /dev/null +++ b/kubespray/contrib/terraform/openstack/modules/ips/main.tf @@ -0,0 +1,46 @@ +resource "null_resource" "dummy_dependency" { + triggers = { + dependency_id = var.router_id + } + depends_on = [ + var.router_internal_port_id + ] +} + +# If user specifies pre-existing IPs to use in k8s_master_fips, do not create new ones. +resource "openstack_networking_floatingip_v2" "k8s_master" { + count = length(var.k8s_master_fips) > 0 ? 0 : var.number_of_k8s_masters + pool = var.floatingip_pool + depends_on = [null_resource.dummy_dependency] +} + +resource "openstack_networking_floatingip_v2" "k8s_masters" { + for_each = var.number_of_k8s_masters == 0 && var.number_of_k8s_masters_no_etcd == 0 ? { for key, value in var.k8s_masters : key => value if value.floating_ip } : {} + pool = var.floatingip_pool + depends_on = [null_resource.dummy_dependency] +} + +# If user specifies pre-existing IPs to use in k8s_master_fips, do not create new ones. +resource "openstack_networking_floatingip_v2" "k8s_master_no_etcd" { + count = length(var.k8s_master_fips) > 0 ? 0 : var.number_of_k8s_masters_no_etcd + pool = var.floatingip_pool + depends_on = [null_resource.dummy_dependency] +} + +resource "openstack_networking_floatingip_v2" "k8s_node" { + count = var.number_of_k8s_nodes + pool = var.floatingip_pool + depends_on = [null_resource.dummy_dependency] +} + +resource "openstack_networking_floatingip_v2" "bastion" { + count = length(var.bastion_fips) > 0 ? 0 : var.number_of_bastions + pool = var.floatingip_pool + depends_on = [null_resource.dummy_dependency] +} + +resource "openstack_networking_floatingip_v2" "k8s_nodes" { + for_each = var.number_of_k8s_nodes == 0 ? { for key, value in var.k8s_nodes : key => value if value.floating_ip } : {} + pool = var.floatingip_pool + depends_on = [null_resource.dummy_dependency] +} diff --git a/kubespray/contrib/terraform/openstack/modules/ips/outputs.tf b/kubespray/contrib/terraform/openstack/modules/ips/outputs.tf new file mode 100644 index 0000000..3ff4622 --- /dev/null +++ b/kubespray/contrib/terraform/openstack/modules/ips/outputs.tf @@ -0,0 +1,25 @@ +# If k8s_master_fips is already defined as input, keep the same value since new FIPs have not been created. +output "k8s_master_fips" { + value = length(var.k8s_master_fips) > 0 ? var.k8s_master_fips : openstack_networking_floatingip_v2.k8s_master[*].address +} + +output "k8s_masters_fips" { + value = openstack_networking_floatingip_v2.k8s_masters +} + +# If k8s_master_fips is already defined as input, keep the same value since new FIPs have not been created. +output "k8s_master_no_etcd_fips" { + value = length(var.k8s_master_fips) > 0 ? var.k8s_master_fips : openstack_networking_floatingip_v2.k8s_master_no_etcd[*].address +} + +output "k8s_node_fips" { + value = openstack_networking_floatingip_v2.k8s_node[*].address +} + +output "k8s_nodes_fips" { + value = openstack_networking_floatingip_v2.k8s_nodes +} + +output "bastion_fips" { + value = length(var.bastion_fips) > 0 ? var.bastion_fips : openstack_networking_floatingip_v2.bastion[*].address +} diff --git a/kubespray/contrib/terraform/openstack/modules/ips/variables.tf b/kubespray/contrib/terraform/openstack/modules/ips/variables.tf new file mode 100644 index 0000000..b52888b --- /dev/null +++ b/kubespray/contrib/terraform/openstack/modules/ips/variables.tf @@ -0,0 +1,27 @@ +variable "number_of_k8s_masters" {} + +variable "number_of_k8s_masters_no_etcd" {} + +variable "number_of_k8s_nodes" {} + +variable "floatingip_pool" {} + +variable "number_of_bastions" {} + +variable "external_net" {} + +variable "network_name" {} + +variable "router_id" { + default = "" +} + +variable "k8s_masters" {} + +variable "k8s_nodes" {} + +variable "k8s_master_fips" {} + +variable "bastion_fips" {} + +variable "router_internal_port_id" {} diff --git a/kubespray/contrib/terraform/openstack/modules/ips/versions.tf b/kubespray/contrib/terraform/openstack/modules/ips/versions.tf new file mode 100644 index 0000000..b7bf5a9 --- /dev/null +++ b/kubespray/contrib/terraform/openstack/modules/ips/versions.tf @@ -0,0 +1,11 @@ +terraform { + required_providers { + null = { + source = "hashicorp/null" + } + openstack = { + source = "terraform-provider-openstack/openstack" + } + } + required_version = ">= 0.12.26" +} diff --git a/kubespray/contrib/terraform/openstack/modules/network/main.tf b/kubespray/contrib/terraform/openstack/modules/network/main.tf new file mode 100644 index 0000000..a6324d7 --- /dev/null +++ b/kubespray/contrib/terraform/openstack/modules/network/main.tf @@ -0,0 +1,34 @@ +resource "openstack_networking_router_v2" "k8s" { + name = "${var.cluster_name}-router" + count = var.use_neutron == 1 && var.router_id == null ? 1 : 0 + admin_state_up = "true" + external_network_id = var.external_net +} + +data "openstack_networking_router_v2" "k8s" { + router_id = var.router_id + count = var.use_neutron == 1 && var.router_id != null ? 1 : 0 +} + +resource "openstack_networking_network_v2" "k8s" { + name = var.network_name + count = var.use_neutron + dns_domain = var.network_dns_domain != null ? var.network_dns_domain : null + admin_state_up = "true" + port_security_enabled = var.port_security_enabled +} + +resource "openstack_networking_subnet_v2" "k8s" { + name = "${var.cluster_name}-internal-network" + count = var.use_neutron + network_id = openstack_networking_network_v2.k8s[count.index].id + cidr = var.subnet_cidr + ip_version = 4 + dns_nameservers = var.dns_nameservers +} + +resource "openstack_networking_router_interface_v2" "k8s" { + count = var.use_neutron + router_id = "%{if openstack_networking_router_v2.k8s != []}${openstack_networking_router_v2.k8s[count.index].id}%{else}${var.router_id}%{endif}" + subnet_id = openstack_networking_subnet_v2.k8s[count.index].id +} diff --git a/kubespray/contrib/terraform/openstack/modules/network/outputs.tf b/kubespray/contrib/terraform/openstack/modules/network/outputs.tf new file mode 100644 index 0000000..0e8a500 --- /dev/null +++ b/kubespray/contrib/terraform/openstack/modules/network/outputs.tf @@ -0,0 +1,15 @@ +output "router_id" { + value = "%{if var.use_neutron == 1} ${var.router_id == null ? element(concat(openstack_networking_router_v2.k8s.*.id, [""]), 0) : var.router_id} %{else} %{endif}" +} + +output "network_id" { + value = element(concat(openstack_networking_network_v2.k8s.*.id, [""]),0) +} + +output "router_internal_port_id" { + value = element(concat(openstack_networking_router_interface_v2.k8s.*.id, [""]), 0) +} + +output "subnet_id" { + value = element(concat(openstack_networking_subnet_v2.k8s.*.id, [""]), 0) +} diff --git a/kubespray/contrib/terraform/openstack/modules/network/variables.tf b/kubespray/contrib/terraform/openstack/modules/network/variables.tf new file mode 100644 index 0000000..6cd7ff7 --- /dev/null +++ b/kubespray/contrib/terraform/openstack/modules/network/variables.tf @@ -0,0 +1,21 @@ +variable "external_net" {} + +variable "network_name" {} + +variable "network_dns_domain" {} + +variable "cluster_name" {} + +variable "dns_nameservers" { + type = list +} + +variable "port_security_enabled" { + type = bool +} + +variable "subnet_cidr" {} + +variable "use_neutron" {} + +variable "router_id" {} diff --git a/kubespray/contrib/terraform/openstack/modules/network/versions.tf b/kubespray/contrib/terraform/openstack/modules/network/versions.tf new file mode 100644 index 0000000..6c94279 --- /dev/null +++ b/kubespray/contrib/terraform/openstack/modules/network/versions.tf @@ -0,0 +1,8 @@ +terraform { + required_providers { + openstack = { + source = "terraform-provider-openstack/openstack" + } + } + required_version = ">= 0.12.26" +} diff --git a/kubespray/contrib/terraform/openstack/sample-inventory/cluster.tfvars b/kubespray/contrib/terraform/openstack/sample-inventory/cluster.tfvars new file mode 100644 index 0000000..8ab7c6d --- /dev/null +++ b/kubespray/contrib/terraform/openstack/sample-inventory/cluster.tfvars @@ -0,0 +1,89 @@ +# your Kubernetes cluster name here +cluster_name = "i-didnt-read-the-docs" + +# list of availability zones available in your OpenStack cluster +#az_list = ["nova"] + +# SSH key to use for access to nodes +public_key_path = "~/.ssh/id_rsa.pub" + +# image to use for bastion, masters, standalone etcd instances, and nodes +image = "" + +# user on the node (ex. core on Container Linux, ubuntu on Ubuntu, etc.) +ssh_user = "" + +# 0|1 bastion nodes +number_of_bastions = 0 + +#flavor_bastion = "" + +# standalone etcds +number_of_etcd = 0 + +# masters +number_of_k8s_masters = 1 + +number_of_k8s_masters_no_etcd = 0 + +number_of_k8s_masters_no_floating_ip = 0 + +number_of_k8s_masters_no_floating_ip_no_etcd = 0 + +flavor_k8s_master = "" + +k8s_masters = { + # "master-1" = { + # "az" = "nova" + # "flavor" = "" + # "floating_ip" = true + # "etcd" = true + # }, + # "master-2" = { + # "az" = "nova" + # "flavor" = "" + # "floating_ip" = false + # "etcd" = true + # }, + # "master-3" = { + # "az" = "nova" + # "flavor" = "" + # "floating_ip" = true + # "etcd" = true + # }, +} + + +# nodes +number_of_k8s_nodes = 2 + +number_of_k8s_nodes_no_floating_ip = 4 + +#flavor_k8s_node = "" + +# GlusterFS +# either 0 or more than one +#number_of_gfs_nodes_no_floating_ip = 0 +#gfs_volume_size_in_gb = 150 +# Container Linux does not support GlusterFS +#image_gfs = "" +# May be different from other nodes +#ssh_user_gfs = "ubuntu" +#flavor_gfs_node = "" + +# networking +network_name = "" + +# Use a existing network with the name of network_name. Set to false to create a network with name of network_name. +# use_existing_network = true + +external_net = "" + +subnet_cidr = "" + +floatingip_pool = "" + +bastion_allowed_remote_ips = ["0.0.0.0/0"] + +# Force port security to be null. Some cloud providers do not allow to set port security. +# force_null_port_security = false diff --git a/kubespray/contrib/terraform/openstack/sample-inventory/group_vars b/kubespray/contrib/terraform/openstack/sample-inventory/group_vars new file mode 120000 index 0000000..3735958 --- /dev/null +++ b/kubespray/contrib/terraform/openstack/sample-inventory/group_vars @@ -0,0 +1 @@ +../../../../inventory/sample/group_vars \ No newline at end of file diff --git a/kubespray/contrib/terraform/openstack/variables.tf b/kubespray/contrib/terraform/openstack/variables.tf new file mode 100644 index 0000000..4bb6efb --- /dev/null +++ b/kubespray/contrib/terraform/openstack/variables.tf @@ -0,0 +1,342 @@ +variable "cluster_name" { + default = "example" +} + +variable "az_list" { + description = "List of Availability Zones to use for masters in your OpenStack cluster" + type = list(string) + default = ["nova"] +} + +variable "az_list_node" { + description = "List of Availability Zones to use for nodes in your OpenStack cluster" + type = list(string) + default = ["nova"] +} + +variable "number_of_bastions" { + default = 1 +} + +variable "number_of_k8s_masters" { + default = 2 +} + +variable "number_of_k8s_masters_no_etcd" { + default = 2 +} + +variable "number_of_etcd" { + default = 2 +} + +variable "number_of_k8s_masters_no_floating_ip" { + default = 2 +} + +variable "number_of_k8s_masters_no_floating_ip_no_etcd" { + default = 2 +} + +variable "number_of_k8s_nodes" { + default = 1 +} + +variable "number_of_k8s_nodes_no_floating_ip" { + default = 1 +} + +variable "number_of_gfs_nodes_no_floating_ip" { + default = 0 +} + +variable "bastion_root_volume_size_in_gb" { + default = 0 +} + +variable "etcd_root_volume_size_in_gb" { + default = 0 +} + +variable "master_root_volume_size_in_gb" { + default = 0 +} + +variable "node_root_volume_size_in_gb" { + default = 0 +} + +variable "gfs_root_volume_size_in_gb" { + default = 0 +} + +variable "gfs_volume_size_in_gb" { + default = 75 +} + +variable "master_volume_type" { + default = "Default" +} + +variable "node_volume_type" { + default = "Default" +} + +variable "public_key_path" { + description = "The path of the ssh pub key" + default = "~/.ssh/id_rsa.pub" +} + +variable "image" { + description = "the image to use" + default = "" +} + +variable "image_gfs" { + description = "Glance image to use for GlusterFS" + default = "" +} + +variable "ssh_user" { + description = "used to fill out tags for ansible inventory" + default = "ubuntu" +} + +variable "ssh_user_gfs" { + description = "used to fill out tags for ansible inventory" + default = "ubuntu" +} + +variable "flavor_bastion" { + description = "Use 'openstack flavor list' command to see what your OpenStack instance uses for IDs" + default = 3 +} + +variable "flavor_k8s_master" { + description = "Use 'openstack flavor list' command to see what your OpenStack instance uses for IDs" + default = 3 +} + +variable "flavor_k8s_node" { + description = "Use 'openstack flavor list' command to see what your OpenStack instance uses for IDs" + default = 3 +} + +variable "flavor_etcd" { + description = "Use 'openstack flavor list' command to see what your OpenStack instance uses for IDs" + default = 3 +} + +variable "flavor_gfs_node" { + description = "Use 'openstack flavor list' command to see what your OpenStack instance uses for IDs" + default = 3 +} + +variable "network_name" { + description = "name of the internal network to use" + default = "internal" +} + +variable "use_existing_network" { + description = "Use an existing network" + type = bool + default = "false" +} + +variable "network_dns_domain" { + description = "dns_domain for the internal network" + type = string + default = null +} + +variable "use_neutron" { + description = "Use neutron" + default = 1 +} + +variable "port_security_enabled" { + description = "Enable port security on the internal network" + type = bool + default = "true" +} + +variable "force_null_port_security" { + description = "Force port security to be null. Some providers does not allow setting port security" + type = bool + default = "false" +} + +variable "subnet_cidr" { + description = "Subnet CIDR block." + type = string + default = "10.0.0.0/24" +} + +variable "dns_nameservers" { + description = "An array of DNS name server names used by hosts in this subnet." + type = list(string) + default = [] +} + +variable "k8s_master_fips" { + description = "specific pre-existing floating IPs to use for master nodes" + type = list(string) + default = [] +} + +variable "bastion_fips" { + description = "specific pre-existing floating IPs to use for bastion node" + type = list(string) + default = [] +} + +variable "floatingip_pool" { + description = "name of the floating ip pool to use" + default = "external" +} + +variable "wait_for_floatingip" { + description = "Terraform will poll the instance until the floating IP has been associated." + default = "false" +} + +variable "external_net" { + description = "uuid of the external/public network" +} + +variable "supplementary_master_groups" { + description = "supplementary kubespray ansible groups for masters, such kube_node" + default = "" +} + +variable "supplementary_node_groups" { + description = "supplementary kubespray ansible groups for worker nodes, such as kube_ingress" + default = "" +} + +variable "bastion_allowed_remote_ips" { + description = "An array of CIDRs allowed to SSH to hosts" + type = list(string) + default = ["0.0.0.0/0"] +} + +variable "master_allowed_remote_ips" { + description = "An array of CIDRs allowed to access API of masters" + type = list(string) + default = ["0.0.0.0/0"] +} + +variable "k8s_allowed_remote_ips" { + description = "An array of CIDRs allowed to SSH to hosts" + type = list(string) + default = [] +} + +variable "k8s_allowed_egress_ips" { + description = "An array of CIDRs allowed for egress traffic" + type = list(string) + default = ["0.0.0.0/0"] +} + +variable "master_allowed_ports" { + type = list(any) + + default = [] +} + +variable "worker_allowed_ports" { + type = list(any) + + default = [ + { + "protocol" = "tcp" + "port_range_min" = 30000 + "port_range_max" = 32767 + "remote_ip_prefix" = "0.0.0.0/0" + }, + ] +} + +variable "bastion_allowed_ports" { + type = list(any) + + default = [] +} + +variable "use_access_ip" { + default = 1 +} + +variable "master_server_group_policy" { + description = "desired server group policy, e.g. anti-affinity" + default = "" +} + +variable "node_server_group_policy" { + description = "desired server group policy, e.g. anti-affinity" + default = "" +} + +variable "etcd_server_group_policy" { + description = "desired server group policy, e.g. anti-affinity" + default = "" +} + +variable "router_id" { + description = "uuid of an externally defined router to use" + default = null +} + +variable "router_internal_port_id" { + description = "uuid of the port connection our router to our network" + default = null +} + +variable "k8s_masters" { + default = {} +} + +variable "k8s_nodes" { + default = {} +} + +variable "additional_server_groups" { + default = {} + type = map(object({ + policy = string + })) +} + +variable "extra_sec_groups" { + default = false +} + +variable "extra_sec_groups_name" { + default = "custom" +} + +variable "image_uuid" { + description = "uuid of image inside openstack to use" + default = "" +} + +variable "image_gfs_uuid" { + description = "uuid of image to be used on gluster fs nodes. If empty defaults to image_uuid" + default = "" +} + +variable "image_master" { + description = "uuid of image inside openstack to use" + default = "" +} + +variable "image_master_uuid" { + description = "uuid of image to be used on master nodes. If empty defaults to image_uuid" + default = "" +} + +variable "group_vars_path" { + description = "path to the inventory group vars directory" + type = string + default = "./group_vars" +} diff --git a/kubespray/contrib/terraform/openstack/versions.tf b/kubespray/contrib/terraform/openstack/versions.tf new file mode 100644 index 0000000..6e4c104 --- /dev/null +++ b/kubespray/contrib/terraform/openstack/versions.tf @@ -0,0 +1,9 @@ +terraform { + required_providers { + openstack = { + source = "terraform-provider-openstack/openstack" + version = "~> 1.17" + } + } + required_version = ">= 1.3.0" +} diff --git a/kubespray/contrib/terraform/terraform.py b/kubespray/contrib/terraform/terraform.py new file mode 100755 index 0000000..f67b9d8 --- /dev/null +++ b/kubespray/contrib/terraform/terraform.py @@ -0,0 +1,474 @@ +#!/usr/bin/env python3 +# +# Copyright 2015 Cisco Systems, Inc. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# original: https://github.com/CiscoCloud/terraform.py + +"""\ +Dynamic inventory for Terraform - finds all `.tfstate` files below the working +directory and generates an inventory based on them. +""" +import argparse +from collections import defaultdict +import random +from functools import wraps +import json +import os +import re + +VERSION = '0.4.0pre' + + +def tfstates(root=None): + root = root or os.getcwd() + for dirpath, _, filenames in os.walk(root): + for name in filenames: + if os.path.splitext(name)[-1] == '.tfstate': + yield os.path.join(dirpath, name) + +def convert_to_v3_structure(attributes, prefix=''): + """ Convert the attributes from v4 to v3 + Receives a dict and return a dictionary """ + result = {} + if isinstance(attributes, str): + # In the case when we receive a string (e.g. values for security_groups) + return {'{}{}'.format(prefix, random.randint(1,10**10)): attributes} + for key, value in attributes.items(): + if isinstance(value, list): + if len(value): + result['{}{}.#'.format(prefix, key, hash)] = len(value) + for i, v in enumerate(value): + result.update(convert_to_v3_structure(v, '{}{}.{}.'.format(prefix, key, i))) + elif isinstance(value, dict): + result['{}{}.%'.format(prefix, key)] = len(value) + for k, v in value.items(): + result['{}{}.{}'.format(prefix, key, k)] = v + else: + result['{}{}'.format(prefix, key)] = value + return result + +def iterresources(filenames): + for filename in filenames: + with open(filename, 'r') as json_file: + state = json.load(json_file) + tf_version = state['version'] + if tf_version == 3: + for module in state['modules']: + name = module['path'][-1] + for key, resource in module['resources'].items(): + yield name, key, resource + elif tf_version == 4: + # In version 4 the structure changes so we need to iterate + # each instance inside the resource branch. + for resource in state['resources']: + name = resource['provider'].split('.')[-1] + for instance in resource['instances']: + key = "{}.{}".format(resource['type'], resource['name']) + if 'index_key' in instance: + key = "{}.{}".format(key, instance['index_key']) + data = {} + data['type'] = resource['type'] + data['provider'] = resource['provider'] + data['depends_on'] = instance.get('depends_on', []) + data['primary'] = {'attributes': convert_to_v3_structure(instance['attributes'])} + if 'id' in instance['attributes']: + data['primary']['id'] = instance['attributes']['id'] + data['primary']['meta'] = instance['attributes'].get('meta',{}) + yield name, key, data + else: + raise KeyError('tfstate version %d not supported' % tf_version) + + +## READ RESOURCES +PARSERS = {} + + +def _clean_dc(dcname): + # Consul DCs are strictly alphanumeric with underscores and hyphens - + # ensure that the consul_dc attribute meets these requirements. + return re.sub('[^\w_\-]', '-', dcname) + + +def iterhosts(resources): + '''yield host tuples of (name, attributes, groups)''' + for module_name, key, resource in resources: + resource_type, name = key.split('.', 1) + try: + parser = PARSERS[resource_type] + except KeyError: + continue + + yield parser(resource, module_name) + + +def iterips(resources): + '''yield ip tuples of (port_id, ip)''' + for module_name, key, resource in resources: + resource_type, name = key.split('.', 1) + if resource_type == 'openstack_networking_floatingip_associate_v2': + yield openstack_floating_ips(resource) + + +def parses(prefix): + def inner(func): + PARSERS[prefix] = func + return func + + return inner + + +def calculate_mantl_vars(func): + """calculate Mantl vars""" + + @wraps(func) + def inner(*args, **kwargs): + name, attrs, groups = func(*args, **kwargs) + + # attrs + if attrs.get('role', '') == 'control': + attrs['consul_is_server'] = True + else: + attrs['consul_is_server'] = False + + # groups + if attrs.get('publicly_routable', False): + groups.append('publicly_routable') + + return name, attrs, groups + + return inner + + +def _parse_prefix(source, prefix, sep='.'): + for compkey, value in list(source.items()): + try: + curprefix, rest = compkey.split(sep, 1) + except ValueError: + continue + + if curprefix != prefix or rest == '#': + continue + + yield rest, value + + +def parse_attr_list(source, prefix, sep='.'): + attrs = defaultdict(dict) + for compkey, value in _parse_prefix(source, prefix, sep): + idx, key = compkey.split(sep, 1) + attrs[idx][key] = value + + return list(attrs.values()) + + +def parse_dict(source, prefix, sep='.'): + return dict(_parse_prefix(source, prefix, sep)) + + +def parse_list(source, prefix, sep='.'): + return [value for _, value in _parse_prefix(source, prefix, sep)] + + +def parse_bool(string_form): + if type(string_form) is bool: + return string_form + + token = string_form.lower()[0] + + if token == 't': + return True + elif token == 'f': + return False + else: + raise ValueError('could not convert %r to a bool' % string_form) + +def sanitize_groups(groups): + _groups = [] + chars_to_replace = ['+', '-', '=', '.', '/', ' '] + for i in groups: + _i = i + for char in chars_to_replace: + _i = _i.replace(char, '_') + _groups.append(_i) + groups.clear() + groups.extend(_groups) + +@parses('equinix_metal_device') +def equinix_metal_device(resource, tfvars=None): + raw_attrs = resource['primary']['attributes'] + name = raw_attrs['hostname'] + groups = [] + + attrs = { + 'id': raw_attrs['id'], + 'facilities': parse_list(raw_attrs, 'facilities'), + 'hostname': raw_attrs['hostname'], + 'operating_system': raw_attrs['operating_system'], + 'locked': parse_bool(raw_attrs['locked']), + 'tags': parse_list(raw_attrs, 'tags'), + 'plan': raw_attrs['plan'], + 'project_id': raw_attrs['project_id'], + 'state': raw_attrs['state'], + # ansible + 'ansible_host': raw_attrs['network.0.address'], + 'ansible_ssh_user': 'root', # Use root by default in metal + # generic + 'ipv4_address': raw_attrs['network.0.address'], + 'public_ipv4': raw_attrs['network.0.address'], + 'ipv6_address': raw_attrs['network.1.address'], + 'public_ipv6': raw_attrs['network.1.address'], + 'private_ipv4': raw_attrs['network.2.address'], + 'provider': 'equinix', + } + + if raw_attrs['operating_system'] == 'flatcar_stable': + # For Flatcar set the ssh_user to core + attrs.update({'ansible_ssh_user': 'core'}) + + # add groups based on attrs + groups.append('equinix_metal_operating_system_%s' % attrs['operating_system']) + groups.append('equinix_metal_locked_%s' % attrs['locked']) + groups.append('equinix_metal_state_%s' % attrs['state']) + groups.append('equinix_metal_plan_%s' % attrs['plan']) + + # groups specific to kubespray + groups = groups + attrs['tags'] + sanitize_groups(groups) + + return name, attrs, groups + + +def openstack_floating_ips(resource): + raw_attrs = resource['primary']['attributes'] + attrs = { + 'ip': raw_attrs['floating_ip'], + 'port_id': raw_attrs['port_id'], + } + return attrs + +def openstack_floating_ips(resource): + raw_attrs = resource['primary']['attributes'] + return raw_attrs['port_id'], raw_attrs['floating_ip'] + +@parses('openstack_compute_instance_v2') +@calculate_mantl_vars +def openstack_host(resource, module_name): + raw_attrs = resource['primary']['attributes'] + name = raw_attrs['name'] + groups = [] + + attrs = { + 'access_ip_v4': raw_attrs['access_ip_v4'], + 'access_ip_v6': raw_attrs['access_ip_v6'], + 'access_ip': raw_attrs['access_ip_v4'], + 'ip': raw_attrs['network.0.fixed_ip_v4'], + 'flavor': parse_dict(raw_attrs, 'flavor', + sep='_'), + 'id': raw_attrs['id'], + 'image': parse_dict(raw_attrs, 'image', + sep='_'), + 'key_pair': raw_attrs['key_pair'], + 'metadata': parse_dict(raw_attrs, 'metadata'), + 'network': parse_attr_list(raw_attrs, 'network'), + 'region': raw_attrs.get('region', ''), + 'security_groups': parse_list(raw_attrs, 'security_groups'), + # workaround for an OpenStack bug where hosts have a different domain + # after they're restarted + 'host_domain': 'novalocal', + 'use_host_domain': True, + # generic + 'public_ipv4': raw_attrs['access_ip_v4'], + 'private_ipv4': raw_attrs['access_ip_v4'], + 'port_id' : raw_attrs['network.0.port'], + 'provider': 'openstack', + } + + if 'floating_ip' in raw_attrs: + attrs['private_ipv4'] = raw_attrs['network.0.fixed_ip_v4'] + + if 'metadata.use_access_ip' in raw_attrs and raw_attrs['metadata.use_access_ip'] == "0": + attrs.pop('access_ip') + + try: + if 'metadata.prefer_ipv6' in raw_attrs and raw_attrs['metadata.prefer_ipv6'] == "1": + attrs.update({ + 'ansible_host': re.sub("[\[\]]", "", raw_attrs['access_ip_v6']), + 'publicly_routable': True, + }) + else: + attrs.update({ + 'ansible_host': raw_attrs['access_ip_v4'], + 'publicly_routable': True, + }) + except (KeyError, ValueError): + attrs.update({'ansible_host': '', 'publicly_routable': False}) + + # Handling of floating IPs has changed: https://github.com/terraform-providers/terraform-provider-openstack/blob/master/CHANGELOG.md#010-june-21-2017 + + # attrs specific to Ansible + if 'metadata.ssh_user' in raw_attrs: + attrs['ansible_user'] = raw_attrs['metadata.ssh_user'] + if 'metadata.ssh_port' in raw_attrs: + attrs['ansible_port'] = raw_attrs['metadata.ssh_port'] + + if 'volume.#' in list(raw_attrs.keys()) and int(raw_attrs['volume.#']) > 0: + device_index = 1 + for key, value in list(raw_attrs.items()): + match = re.search("^volume.*.device$", key) + if match: + attrs['disk_volume_device_'+str(device_index)] = value + device_index += 1 + + + # attrs specific to Mantl + attrs.update({ + 'role': attrs['metadata'].get('role', 'none') + }) + + # add groups based on attrs + groups.append('os_image=' + str(attrs['image']['id'])) + groups.append('os_flavor=' + str(attrs['flavor']['name'])) + groups.extend('os_metadata_%s=%s' % item + for item in list(attrs['metadata'].items())) + groups.append('os_region=' + str(attrs['region'])) + + # groups specific to kubespray + for group in attrs['metadata'].get('kubespray_groups', "").split(","): + groups.append(group) + + sanitize_groups(groups) + + return name, attrs, groups + + +def iter_host_ips(hosts, ips): + '''Update hosts that have an entry in the floating IP list''' + for host in hosts: + port_id = host[1]['port_id'] + + if port_id in ips: + ip = ips[port_id] + + host[1].update({ + 'access_ip_v4': ip, + 'access_ip': ip, + 'public_ipv4': ip, + 'ansible_host': ip, + }) + + if 'use_access_ip' in host[1]['metadata'] and host[1]['metadata']['use_access_ip'] == "0": + host[1].pop('access_ip') + + yield host + + +## QUERY TYPES +def query_host(hosts, target): + for name, attrs, _ in hosts: + if name == target: + return attrs + + return {} + + +def query_list(hosts): + groups = defaultdict(dict) + meta = {} + + for name, attrs, hostgroups in hosts: + for group in set(hostgroups): + # Ansible 2.6.2 stopped supporting empty group names: https://github.com/ansible/ansible/pull/42584/commits/d4cd474b42ed23d8f8aabb2a7f84699673852eaf + # Empty group name defaults to "all" in Ansible < 2.6.2 so we alter empty group names to "all" + if not group: group = "all" + + groups[group].setdefault('hosts', []) + groups[group]['hosts'].append(name) + + meta[name] = attrs + + groups['_meta'] = {'hostvars': meta} + return groups + + +def query_hostfile(hosts): + out = ['## begin hosts generated by terraform.py ##'] + out.extend( + '{}\t{}'.format(attrs['ansible_host'].ljust(16), name) + for name, attrs, _ in hosts + ) + + out.append('## end hosts generated by terraform.py ##') + return '\n'.join(out) + + +def main(): + parser = argparse.ArgumentParser( + __file__, __doc__, + formatter_class=argparse.ArgumentDefaultsHelpFormatter, ) + modes = parser.add_mutually_exclusive_group(required=True) + modes.add_argument('--list', + action='store_true', + help='list all variables') + modes.add_argument('--host', help='list variables for a single host') + modes.add_argument('--version', + action='store_true', + help='print version and exit') + modes.add_argument('--hostfile', + action='store_true', + help='print hosts as a /etc/hosts snippet') + parser.add_argument('--pretty', + action='store_true', + help='pretty-print output JSON') + parser.add_argument('--nometa', + action='store_true', + help='with --list, exclude hostvars') + default_root = os.environ.get('TERRAFORM_STATE_ROOT', + os.path.abspath(os.path.join(os.path.dirname(__file__), + '..', '..', ))) + parser.add_argument('--root', + default=default_root, + help='custom root to search for `.tfstate`s in') + + args = parser.parse_args() + + if args.version: + print('%s %s' % (__file__, VERSION)) + parser.exit() + + hosts = iterhosts(iterresources(tfstates(args.root))) + + # Perform a second pass on the file to pick up floating_ip entries to update the ip address of referenced hosts + ips = dict(iterips(iterresources(tfstates(args.root)))) + + if ips: + hosts = iter_host_ips(hosts, ips) + + if args.list: + output = query_list(hosts) + if args.nometa: + del output['_meta'] + print(json.dumps(output, indent=4 if args.pretty else None)) + elif args.host: + output = query_host(hosts, args.host) + print(json.dumps(output, indent=4 if args.pretty else None)) + elif args.hostfile: + output = query_hostfile(hosts) + print(output) + + parser.exit() + + +if __name__ == '__main__': + main() diff --git a/kubespray/contrib/terraform/upcloud/README.md b/kubespray/contrib/terraform/upcloud/README.md new file mode 100644 index 0000000..c893c34 --- /dev/null +++ b/kubespray/contrib/terraform/upcloud/README.md @@ -0,0 +1,143 @@ +# Kubernetes on UpCloud with Terraform + +Provision a Kubernetes cluster on [UpCloud](https://upcloud.com/) using Terraform and Kubespray + +## Overview + +The setup looks like following + +```text + Kubernetes cluster ++--------------------------+ +| +--------------+ | +| | +--------------+ | +| --> | | | | +| | | Master/etcd | | +| | | node(s) | | +| +-+ | | +| +--------------+ | +| ^ | +| | | +| v | +| +--------------+ | +| | +--------------+ | +| --> | | | | +| | | Worker | | +| | | node(s) | | +| +-+ | | +| +--------------+ | ++--------------------------+ +``` + +The nodes uses a private network for node to node communication and a public interface for all external communication. + +## Requirements + +* Terraform 0.13.0 or newer + +## Quickstart + +NOTE: Assumes you are at the root of the kubespray repo. + +For authentication in your cluster you can use the environment variables. + +```bash +export TF_VAR_UPCLOUD_USERNAME=username +export TF_VAR_UPCLOUD_PASSWORD=password +``` + +To allow API access to your UpCloud account, you need to allow API connections by visiting [Account-page](https://hub.upcloud.com/account) in your UpCloud Hub. + +Copy the cluster configuration file. + +```bash +CLUSTER=my-upcloud-cluster +cp -r inventory/sample inventory/$CLUSTER +cp contrib/terraform/upcloud/cluster-settings.tfvars inventory/$CLUSTER/ +export ANSIBLE_CONFIG=ansible.cfg +cd inventory/$CLUSTER +``` + +Edit `cluster-settings.tfvars` to match your requirement. + +Run Terraform to create the infrastructure. + +```bash +terraform init ../../contrib/terraform/upcloud +terraform apply --var-file cluster-settings.tfvars \ + -state=tfstate-$CLUSTER.tfstate \ + ../../contrib/terraform/upcloud/ +``` + +You should now have a inventory file named `inventory.ini` that you can use with kubespray. +You can use the inventory file with kubespray to set up a cluster. + +It is a good idea to check that you have basic SSH connectivity to the nodes. You can do that by: + +```bash +ansible -i inventory.ini -m ping all +``` + +You can setup Kubernetes with kubespray using the generated inventory: + +```bash +ansible-playbook -i inventory.ini ../../cluster.yml -b -v +``` + +## Teardown + +You can teardown your infrastructure using the following Terraform command: + +```bash +terraform destroy --var-file cluster-settings.tfvars \ + -state=tfstate-$CLUSTER.tfstate \ + ../../contrib/terraform/upcloud/ +``` + +## Variables + +* `prefix`: Prefix to add to all resources, if set to "" don't set any prefix +* `template_name`: The name or UUID of a base image +* `username`: a user to access the nodes, defaults to "ubuntu" +* `private_network_cidr`: CIDR to use for the private network, defaults to "172.16.0.0/24" +* `ssh_public_keys`: List of public SSH keys to install on all machines +* `zone`: The zone where to run the cluster +* `machines`: Machines to provision. Key of this object will be used as the name of the machine + * `node_type`: The role of this node *(master|worker)* + * `plan`: Preconfigured cpu/mem plan to use (disables `cpu` and `mem` attributes below) + * `cpu`: number of cpu cores + * `mem`: memory size in MB + * `disk_size`: The size of the storage in GB + * `additional_disks`: Additional disks to attach to the node. + * `size`: The size of the additional disk in GB + * `tier`: The tier of disk to use (`maxiops` is the only one you can choose atm) +* `firewall_enabled`: Enable firewall rules +* `firewall_default_deny_in`: Set the firewall to deny inbound traffic by default. Automatically adds UpCloud DNS server and NTP port allowlisting. +* `firewall_default_deny_out`: Set the firewall to deny outbound traffic by default. +* `master_allowed_remote_ips`: List of IP ranges that should be allowed to access API of masters + * `start_address`: Start of address range to allow + * `end_address`: End of address range to allow +* `k8s_allowed_remote_ips`: List of IP ranges that should be allowed SSH access to all nodes + * `start_address`: Start of address range to allow + * `end_address`: End of address range to allow +* `master_allowed_ports`: List of port ranges that should be allowed to access the masters + * `protocol`: Protocol *(tcp|udp|icmp)* + * `port_range_min`: Start of port range to allow + * `port_range_max`: End of port range to allow + * `start_address`: Start of address range to allow + * `end_address`: End of address range to allow +* `worker_allowed_ports`: List of port ranges that should be allowed to access the workers + * `protocol`: Protocol *(tcp|udp|icmp)* + * `port_range_min`: Start of port range to allow + * `port_range_max`: End of port range to allow + * `start_address`: Start of address range to allow + * `end_address`: End of address range to allow +* `loadbalancer_enabled`: Enable managed load balancer +* `loadbalancer_plan`: Plan to use for load balancer *(development|production-small)* +* `loadbalancers`: Ports to load balance and which machines to forward to. Key of this object will be used as the name of the load balancer frontends/backends + * `port`: Port to load balance. + * `target_port`: Port to the backend servers. + * `backend_servers`: List of servers that traffic to the port should be forwarded to. +* `server_groups`: Group servers together + * `servers`: The servers that should be included in the group. + * `anti_affinity`: If anti-affinity should be enabled, try to spread the VMs out on separate nodes. diff --git a/kubespray/contrib/terraform/upcloud/cluster-settings.tfvars b/kubespray/contrib/terraform/upcloud/cluster-settings.tfvars new file mode 100644 index 0000000..d88945f --- /dev/null +++ b/kubespray/contrib/terraform/upcloud/cluster-settings.tfvars @@ -0,0 +1,148 @@ +# See: https://developers.upcloud.com/1.3/5-zones/ +zone = "fi-hel1" +username = "ubuntu" + +# Prefix to use for all resources to separate them from other resources +prefix = "kubespray" + +inventory_file = "inventory.ini" + +# Set the operating system using UUID or exact name +template_name = "Ubuntu Server 20.04 LTS (Focal Fossa)" + +ssh_public_keys = [ + # Put your public SSH key here + "ssh-rsa public key 1", + "ssh-rsa public key 2", +] + +# check list of available plan https://developers.upcloud.com/1.3/7-plans/ +machines = { + "master-0" : { + "node_type" : "master", + # plan to use instead of custom cpu/mem + "plan" : null, + #number of cpu cores + "cpu" : "2", + #memory size in MB + "mem" : "4096" + # The size of the storage in GB + "disk_size" : 250 + "additional_disks" : {} + }, + "worker-0" : { + "node_type" : "worker", + # plan to use instead of custom cpu/mem + "plan" : null, + #number of cpu cores + "cpu" : "2", + #memory size in MB + "mem" : "4096" + # The size of the storage in GB + "disk_size" : 250 + "additional_disks" : { + # "some-disk-name-1": { + # "size": 100, + # "tier": "maxiops", + # }, + # "some-disk-name-2": { + # "size": 100, + # "tier": "maxiops", + # } + } + }, + "worker-1" : { + "node_type" : "worker", + # plan to use instead of custom cpu/mem + "plan" : null, + #number of cpu cores + "cpu" : "2", + #memory size in MB + "mem" : "4096" + # The size of the storage in GB + "disk_size" : 250 + "additional_disks" : { + # "some-disk-name-1": { + # "size": 100, + # "tier": "maxiops", + # }, + # "some-disk-name-2": { + # "size": 100, + # "tier": "maxiops", + # } + } + }, + "worker-2" : { + "node_type" : "worker", + # plan to use instead of custom cpu/mem + "plan" : null, + #number of cpu cores + "cpu" : "2", + #memory size in MB + "mem" : "4096" + # The size of the storage in GB + "disk_size" : 250 + "additional_disks" : { + # "some-disk-name-1": { + # "size": 100, + # "tier": "maxiops", + # }, + # "some-disk-name-2": { + # "size": 100, + # "tier": "maxiops", + # } + } + } +} + +firewall_enabled = false +firewall_default_deny_in = false +firewall_default_deny_out = false + +master_allowed_remote_ips = [ + { + "start_address" : "0.0.0.0" + "end_address" : "255.255.255.255" + } +] + +k8s_allowed_remote_ips = [ + { + "start_address" : "0.0.0.0" + "end_address" : "255.255.255.255" + } +] + +master_allowed_ports = [] +worker_allowed_ports = [] + +loadbalancer_enabled = false +loadbalancer_plan = "development" +loadbalancers = { + # "http" : { + # "port" : 80, + # "target_port" : 80, + # "backend_servers" : [ + # "worker-0", + # "worker-1", + # "worker-2" + # ] + # } +} + +server_groups = { + # "control-plane" = { + # servers = [ + # "master-0" + # ] + # anti_affinity = true + # }, + # "workers" = { + # servers = [ + # "worker-0", + # "worker-1", + # "worker-2" + # ] + # anti_affinity = true + # } +} \ No newline at end of file diff --git a/kubespray/contrib/terraform/upcloud/main.tf b/kubespray/contrib/terraform/upcloud/main.tf new file mode 100644 index 0000000..93483a9 --- /dev/null +++ b/kubespray/contrib/terraform/upcloud/main.tf @@ -0,0 +1,73 @@ + +terraform { + required_version = ">= 0.13.0" +} +provider "upcloud" { + # Your UpCloud credentials are read from environment variables: + username = var.UPCLOUD_USERNAME + password = var.UPCLOUD_PASSWORD +} + +module "kubernetes" { + source = "./modules/kubernetes-cluster" + + prefix = var.prefix + zone = var.zone + + template_name = var.template_name + username = var.username + + private_network_cidr = var.private_network_cidr + + machines = var.machines + + ssh_public_keys = var.ssh_public_keys + + firewall_enabled = var.firewall_enabled + firewall_default_deny_in = var.firewall_default_deny_in + firewall_default_deny_out = var.firewall_default_deny_out + master_allowed_remote_ips = var.master_allowed_remote_ips + k8s_allowed_remote_ips = var.k8s_allowed_remote_ips + master_allowed_ports = var.master_allowed_ports + worker_allowed_ports = var.worker_allowed_ports + + loadbalancer_enabled = var.loadbalancer_enabled + loadbalancer_plan = var.loadbalancer_plan + loadbalancers = var.loadbalancers + + server_groups = var.server_groups +} + +# +# Generate ansible inventory +# + +data "template_file" "inventory" { + template = file("${path.module}/templates/inventory.tpl") + + vars = { + connection_strings_master = join("\n", formatlist("%s ansible_user=ubuntu ansible_host=%s ip=%s etcd_member_name=etcd%d", + keys(module.kubernetes.master_ip), + values(module.kubernetes.master_ip).*.public_ip, + values(module.kubernetes.master_ip).*.private_ip, + range(1, length(module.kubernetes.master_ip) + 1))) + connection_strings_worker = join("\n", formatlist("%s ansible_user=ubuntu ansible_host=%s ip=%s", + keys(module.kubernetes.worker_ip), + values(module.kubernetes.worker_ip).*.public_ip, + values(module.kubernetes.worker_ip).*.private_ip)) + list_master = join("\n", formatlist("%s", + keys(module.kubernetes.master_ip))) + list_worker = join("\n", formatlist("%s", + keys(module.kubernetes.worker_ip))) + } +} + +resource "null_resource" "inventories" { + provisioner "local-exec" { + command = "echo '${data.template_file.inventory.rendered}' > ${var.inventory_file}" + } + + triggers = { + template = data.template_file.inventory.rendered + } +} diff --git a/kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/main.tf b/kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/main.tf new file mode 100644 index 0000000..c2d43a3 --- /dev/null +++ b/kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/main.tf @@ -0,0 +1,558 @@ +locals { + # Create a list of all disks to create + disks = flatten([ + for node_name, machine in var.machines : [ + for disk_name, disk in machine.additional_disks : { + disk = disk + disk_name = disk_name + node_name = node_name + } + ] + ]) + + lb_backend_servers = flatten([ + for lb_name, loadbalancer in var.loadbalancers : [ + for backend_server in loadbalancer.backend_servers : { + port = loadbalancer.target_port + lb_name = lb_name + server_name = backend_server + } + ] + ]) + + # If prefix is set, all resources will be prefixed with "${var.prefix}-" + # Else don't prefix with anything + resource-prefix = "%{ if var.prefix != ""}${var.prefix}-%{ endif }" +} + +resource "upcloud_network" "private" { + name = "${local.resource-prefix}k8s-network" + zone = var.zone + + ip_network { + address = var.private_network_cidr + dhcp = true + family = "IPv4" + } +} + +resource "upcloud_storage" "additional_disks" { + for_each = { + for disk in local.disks: "${disk.node_name}_${disk.disk_name}" => disk.disk + } + + size = each.value.size + tier = each.value.tier + title = "${local.resource-prefix}${each.key}" + zone = var.zone +} + +resource "upcloud_server" "master" { + for_each = { + for name, machine in var.machines : + name => machine + if machine.node_type == "master" + } + + hostname = "${local.resource-prefix}${each.key}" + plan = each.value.plan + cpu = each.value.plan == null ? each.value.cpu : null + mem = each.value.plan == null ? each.value.mem : null + zone = var.zone + + template { + storage = var.template_name + size = each.value.disk_size + } + + # Public network interface + network_interface { + type = "public" + } + + # Private network interface + network_interface { + type = "private" + network = upcloud_network.private.id + } + + # Ignore volumes created by csi-driver + lifecycle { + ignore_changes = [storage_devices] + } + + firewall = var.firewall_enabled + + dynamic "storage_devices" { + for_each = { + for disk_key_name, disk in upcloud_storage.additional_disks : + disk_key_name => disk + # Only add the disk if it matches the node name in the start of its name + if length(regexall("^${each.key}_.+", disk_key_name)) > 0 + } + + content { + storage = storage_devices.value.id + } + } + + # Include at least one public SSH key + login { + user = var.username + keys = var.ssh_public_keys + create_password = false + } +} + +resource "upcloud_server" "worker" { + for_each = { + for name, machine in var.machines : + name => machine + if machine.node_type == "worker" + } + + hostname = "${local.resource-prefix}${each.key}" + plan = each.value.plan + cpu = each.value.plan == null ? each.value.cpu : null + mem = each.value.plan == null ? each.value.mem : null + zone = var.zone + + template { + storage = var.template_name + size = each.value.disk_size + } + + # Public network interface + network_interface { + type = "public" + } + + # Private network interface + network_interface { + type = "private" + network = upcloud_network.private.id + } + + # Ignore volumes created by csi-driver + lifecycle { + ignore_changes = [storage_devices] + } + + firewall = var.firewall_enabled + + dynamic "storage_devices" { + for_each = { + for disk_key_name, disk in upcloud_storage.additional_disks : + disk_key_name => disk + # Only add the disk if it matches the node name in the start of its name + if length(regexall("^${each.key}_.+", disk_key_name)) > 0 + } + + content { + storage = storage_devices.value.id + } + } + + # Include at least one public SSH key + login { + user = var.username + keys = var.ssh_public_keys + create_password = false + } +} + +resource "upcloud_firewall_rules" "master" { + for_each = upcloud_server.master + server_id = each.value.id + + dynamic firewall_rule { + for_each = var.master_allowed_remote_ips + + content { + action = "accept" + comment = "Allow master API access from this network" + destination_port_end = "6443" + destination_port_start = "6443" + direction = "in" + family = "IPv4" + protocol = "tcp" + source_address_end = firewall_rule.value.end_address + source_address_start = firewall_rule.value.start_address + } + } + + dynamic firewall_rule { + for_each = length(var.master_allowed_remote_ips) > 0 ? [1] : [] + + content { + action = "drop" + comment = "Deny master API access from other networks" + destination_port_end = "6443" + destination_port_start = "6443" + direction = "in" + family = "IPv4" + protocol = "tcp" + source_address_end = "255.255.255.255" + source_address_start = "0.0.0.0" + } + } + + dynamic firewall_rule { + for_each = var.k8s_allowed_remote_ips + + content { + action = "accept" + comment = "Allow SSH from this network" + destination_port_end = "22" + destination_port_start = "22" + direction = "in" + family = "IPv4" + protocol = "tcp" + source_address_end = firewall_rule.value.end_address + source_address_start = firewall_rule.value.start_address + } + } + + dynamic firewall_rule { + for_each = length(var.k8s_allowed_remote_ips) > 0 ? [1] : [] + + content { + action = "drop" + comment = "Deny SSH from other networks" + destination_port_end = "22" + destination_port_start = "22" + direction = "in" + family = "IPv4" + protocol = "tcp" + source_address_end = "255.255.255.255" + source_address_start = "0.0.0.0" + } + } + + dynamic firewall_rule { + for_each = var.master_allowed_ports + + content { + action = "accept" + comment = "Allow access on this port" + destination_port_end = firewall_rule.value.port_range_max + destination_port_start = firewall_rule.value.port_range_min + direction = "in" + family = "IPv4" + protocol = firewall_rule.value.protocol + source_address_end = firewall_rule.value.end_address + source_address_start = firewall_rule.value.start_address + } + } + + dynamic firewall_rule { + for_each = var.firewall_default_deny_in ? ["tcp", "udp"] : [] + + content { + action = "accept" + comment = "UpCloud DNS" + source_port_end = "53" + source_port_start = "53" + direction = "in" + family = "IPv4" + protocol = firewall_rule.value + source_address_end = "94.237.40.9" + source_address_start = "94.237.40.9" + } + } + + dynamic firewall_rule { + for_each = var.firewall_default_deny_in ? ["tcp", "udp"] : [] + + content { + action = "accept" + comment = "UpCloud DNS" + source_port_end = "53" + source_port_start = "53" + direction = "in" + family = "IPv4" + protocol = firewall_rule.value + source_address_end = "94.237.127.9" + source_address_start = "94.237.127.9" + } + } + + dynamic firewall_rule { + for_each = var.firewall_default_deny_in ? ["tcp", "udp"] : [] + + content { + action = "accept" + comment = "UpCloud DNS" + source_port_end = "53" + source_port_start = "53" + direction = "in" + family = "IPv6" + protocol = firewall_rule.value + source_address_end = "2a04:3540:53::1" + source_address_start = "2a04:3540:53::1" + } + } + + dynamic firewall_rule { + for_each = var.firewall_default_deny_in ? ["tcp", "udp"] : [] + + content { + action = "accept" + comment = "UpCloud DNS" + source_port_end = "53" + source_port_start = "53" + direction = "in" + family = "IPv6" + protocol = firewall_rule.value + source_address_end = "2a04:3544:53::1" + source_address_start = "2a04:3544:53::1" + } + } + + dynamic firewall_rule { + for_each = var.firewall_default_deny_in ? ["udp"] : [] + + content { + action = "accept" + comment = "NTP Port" + source_port_end = "123" + source_port_start = "123" + direction = "in" + family = "IPv4" + protocol = firewall_rule.value + source_address_end = "255.255.255.255" + source_address_start = "0.0.0.0" + } + } + + dynamic firewall_rule { + for_each = var.firewall_default_deny_in ? ["udp"] : [] + + content { + action = "accept" + comment = "NTP Port" + source_port_end = "123" + source_port_start = "123" + direction = "in" + family = "IPv6" + protocol = firewall_rule.value + } + } + + firewall_rule { + action = var.firewall_default_deny_in ? "drop" : "accept" + direction = "in" + } + + firewall_rule { + action = var.firewall_default_deny_out ? "drop" : "accept" + direction = "out" + } +} + +resource "upcloud_firewall_rules" "k8s" { + for_each = upcloud_server.worker + server_id = each.value.id + + dynamic firewall_rule { + for_each = var.k8s_allowed_remote_ips + + content { + action = "accept" + comment = "Allow SSH from this network" + destination_port_end = "22" + destination_port_start = "22" + direction = "in" + family = "IPv4" + protocol = "tcp" + source_address_end = firewall_rule.value.end_address + source_address_start = firewall_rule.value.start_address + } + } + + dynamic firewall_rule { + for_each = length(var.k8s_allowed_remote_ips) > 0 ? [1] : [] + + content { + action = "drop" + comment = "Deny SSH from other networks" + destination_port_end = "22" + destination_port_start = "22" + direction = "in" + family = "IPv4" + protocol = "tcp" + source_address_end = "255.255.255.255" + source_address_start = "0.0.0.0" + } + } + + dynamic firewall_rule { + for_each = var.worker_allowed_ports + + content { + action = "accept" + comment = "Allow access on this port" + destination_port_end = firewall_rule.value.port_range_max + destination_port_start = firewall_rule.value.port_range_min + direction = "in" + family = "IPv4" + protocol = firewall_rule.value.protocol + source_address_end = firewall_rule.value.end_address + source_address_start = firewall_rule.value.start_address + } + } + + dynamic firewall_rule { + for_each = var.firewall_default_deny_in ? ["tcp", "udp"] : [] + + content { + action = "accept" + comment = "UpCloud DNS" + source_port_end = "53" + source_port_start = "53" + direction = "in" + family = "IPv4" + protocol = firewall_rule.value + source_address_end = "94.237.40.9" + source_address_start = "94.237.40.9" + } + } + + dynamic firewall_rule { + for_each = var.firewall_default_deny_in ? ["tcp", "udp"] : [] + + content { + action = "accept" + comment = "UpCloud DNS" + source_port_end = "53" + source_port_start = "53" + direction = "in" + family = "IPv4" + protocol = firewall_rule.value + source_address_end = "94.237.127.9" + source_address_start = "94.237.127.9" + } + } + + dynamic firewall_rule { + for_each = var.firewall_default_deny_in ? ["tcp", "udp"] : [] + + content { + action = "accept" + comment = "UpCloud DNS" + source_port_end = "53" + source_port_start = "53" + direction = "in" + family = "IPv6" + protocol = firewall_rule.value + source_address_end = "2a04:3540:53::1" + source_address_start = "2a04:3540:53::1" + } + } + + dynamic firewall_rule { + for_each = var.firewall_default_deny_in ? ["tcp", "udp"] : [] + + content { + action = "accept" + comment = "UpCloud DNS" + source_port_end = "53" + source_port_start = "53" + direction = "in" + family = "IPv6" + protocol = firewall_rule.value + source_address_end = "2a04:3544:53::1" + source_address_start = "2a04:3544:53::1" + } + } + + dynamic firewall_rule { + for_each = var.firewall_default_deny_in ? ["udp"] : [] + + content { + action = "accept" + comment = "NTP Port" + source_port_end = "123" + source_port_start = "123" + direction = "in" + family = "IPv4" + protocol = firewall_rule.value + source_address_end = "255.255.255.255" + source_address_start = "0.0.0.0" + } + } + + dynamic firewall_rule { + for_each = var.firewall_default_deny_in ? ["udp"] : [] + + content { + action = "accept" + comment = "NTP Port" + source_port_end = "123" + source_port_start = "123" + direction = "in" + family = "IPv6" + protocol = firewall_rule.value + } + } + + firewall_rule { + action = var.firewall_default_deny_in ? "drop" : "accept" + direction = "in" + } + + firewall_rule { + action = var.firewall_default_deny_out ? "drop" : "accept" + direction = "out" + } +} + +resource "upcloud_loadbalancer" "lb" { + count = var.loadbalancer_enabled ? 1 : 0 + configured_status = "started" + name = "${local.resource-prefix}lb" + plan = var.loadbalancer_plan + zone = var.zone + network = upcloud_network.private.id +} + +resource "upcloud_loadbalancer_backend" "lb_backend" { + for_each = var.loadbalancer_enabled ? var.loadbalancers : {} + + loadbalancer = upcloud_loadbalancer.lb[0].id + name = "lb-backend-${each.key}" +} + +resource "upcloud_loadbalancer_frontend" "lb_frontend" { + for_each = var.loadbalancer_enabled ? var.loadbalancers : {} + + loadbalancer = upcloud_loadbalancer.lb[0].id + name = "lb-frontend-${each.key}" + mode = "tcp" + port = each.value.port + default_backend_name = upcloud_loadbalancer_backend.lb_backend[each.key].name +} + +resource "upcloud_loadbalancer_static_backend_member" "lb_backend_member" { + for_each = { + for be_server in local.lb_backend_servers: + "${be_server.server_name}-lb-backend-${be_server.lb_name}" => be_server + if var.loadbalancer_enabled + } + + backend = upcloud_loadbalancer_backend.lb_backend[each.value.lb_name].id + name = "${local.resource-prefix}${each.key}" + ip = merge(upcloud_server.master, upcloud_server.worker)[each.value.server_name].network_interface[1].ip_address + port = each.value.port + weight = 100 + max_sessions = var.loadbalancer_plan == "production-small" ? 50000 : 1000 + enabled = true +} + +resource "upcloud_server_group" "server_groups" { + for_each = var.server_groups + title = each.key + anti_affinity = each.value.anti_affinity + labels = {} + members = [for server in each.value.servers : merge(upcloud_server.master, upcloud_server.worker)[server].id] +} \ No newline at end of file diff --git a/kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/output.tf b/kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/output.tf new file mode 100644 index 0000000..c1f8c7c --- /dev/null +++ b/kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/output.tf @@ -0,0 +1,24 @@ + +output "master_ip" { + value = { + for instance in upcloud_server.master : + instance.hostname => { + "public_ip": instance.network_interface[0].ip_address + "private_ip": instance.network_interface[1].ip_address + } + } +} + +output "worker_ip" { + value = { + for instance in upcloud_server.worker : + instance.hostname => { + "public_ip": instance.network_interface[0].ip_address + "private_ip": instance.network_interface[1].ip_address + } + } +} + +output "loadbalancer_domain" { + value = var.loadbalancer_enabled ? upcloud_loadbalancer.lb[0].dns_name : null +} diff --git a/kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/variables.tf b/kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/variables.tf new file mode 100644 index 0000000..8c492ae --- /dev/null +++ b/kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/variables.tf @@ -0,0 +1,105 @@ +variable "prefix" { + type = string +} + +variable "zone" { + type = string +} + +variable "template_name" {} + +variable "username" {} + +variable "private_network_cidr" {} + +variable "machines" { + description = "Cluster machines" + type = map(object({ + node_type = string + plan = string + cpu = string + mem = string + disk_size = number + additional_disks = map(object({ + size = number + tier = string + })) + })) +} + +variable "ssh_public_keys" { + type = list(string) +} + +variable "firewall_enabled" { + type = bool +} + +variable "master_allowed_remote_ips" { + type = list(object({ + start_address = string + end_address = string + })) +} + +variable "k8s_allowed_remote_ips" { + type = list(object({ + start_address = string + end_address = string + })) +} + +variable "master_allowed_ports" { + type = list(object({ + protocol = string + port_range_min = number + port_range_max = number + start_address = string + end_address = string + })) +} + +variable "worker_allowed_ports" { + type = list(object({ + protocol = string + port_range_min = number + port_range_max = number + start_address = string + end_address = string + })) +} + +variable "firewall_default_deny_in" { + type = bool +} + +variable "firewall_default_deny_out" { + type = bool +} + +variable "loadbalancer_enabled" { + type = bool +} + +variable "loadbalancer_plan" { + type = string +} + +variable "loadbalancers" { + description = "Load balancers" + + type = map(object({ + port = number + target_port = number + backend_servers = list(string) + })) +} + +variable "server_groups" { + description = "Server groups" + + type = map(object({ + anti_affinity = bool + servers = list(string) + })) +} \ No newline at end of file diff --git a/kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/versions.tf b/kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/versions.tf new file mode 100644 index 0000000..75230b9 --- /dev/null +++ b/kubespray/contrib/terraform/upcloud/modules/kubernetes-cluster/versions.tf @@ -0,0 +1,10 @@ + +terraform { + required_providers { + upcloud = { + source = "UpCloudLtd/upcloud" + version = "~>2.7.1" + } + } + required_version = ">= 0.13" +} diff --git a/kubespray/contrib/terraform/upcloud/output.tf b/kubespray/contrib/terraform/upcloud/output.tf new file mode 100644 index 0000000..006e3b1 --- /dev/null +++ b/kubespray/contrib/terraform/upcloud/output.tf @@ -0,0 +1,12 @@ + +output "master_ip" { + value = module.kubernetes.master_ip +} + +output "worker_ip" { + value = module.kubernetes.worker_ip +} + +output "loadbalancer_domain" { + value = module.kubernetes.loadbalancer_domain +} diff --git a/kubespray/contrib/terraform/upcloud/sample-inventory/cluster.tfvars b/kubespray/contrib/terraform/upcloud/sample-inventory/cluster.tfvars new file mode 100644 index 0000000..4e8ade8 --- /dev/null +++ b/kubespray/contrib/terraform/upcloud/sample-inventory/cluster.tfvars @@ -0,0 +1,149 @@ +# See: https://developers.upcloud.com/1.3/5-zones/ +zone = "fi-hel1" +username = "ubuntu" + +# Prefix to use for all resources to separate them from other resources +prefix = "kubespray" + +inventory_file = "inventory.ini" + +# Set the operating system using UUID or exact name +template_name = "Ubuntu Server 20.04 LTS (Focal Fossa)" + +ssh_public_keys = [ + # Put your public SSH key here + "ssh-rsa I-did-not-read-the-docs", + "ssh-rsa I-did-not-read-the-docs 2", +] + +# check list of available plan https://developers.upcloud.com/1.3/7-plans/ +machines = { + "master-0" : { + "node_type" : "master", + # plan to use instead of custom cpu/mem + "plan" : null, + #number of cpu cores + "cpu" : "2", + #memory size in MB + "mem" : "4096" + # The size of the storage in GB + "disk_size" : 250 + "additional_disks": {} + }, + "worker-0" : { + "node_type" : "worker", + # plan to use instead of custom cpu/mem + "plan" : null, + #number of cpu cores + "cpu" : "2", + #memory size in MB + "mem" : "4096" + # The size of the storage in GB + "disk_size" : 250 + "additional_disks": { + # "some-disk-name-1": { + # "size": 100, + # "tier": "maxiops", + # }, + # "some-disk-name-2": { + # "size": 100, + # "tier": "maxiops", + # } + } + }, + "worker-1" : { + "node_type" : "worker", + # plan to use instead of custom cpu/mem + "plan" : null, + #number of cpu cores + "cpu" : "2", + #memory size in MB + "mem" : "4096" + # The size of the storage in GB + "disk_size" : 250 + "additional_disks": { + # "some-disk-name-1": { + # "size": 100, + # "tier": "maxiops", + # }, + # "some-disk-name-2": { + # "size": 100, + # "tier": "maxiops", + # } + } + }, + "worker-2" : { + "node_type" : "worker", + # plan to use instead of custom cpu/mem + "plan" : null, + #number of cpu cores + "cpu" : "2", + #memory size in MB + "mem" : "4096" + # The size of the storage in GB + "disk_size" : 250 + "additional_disks": { + # "some-disk-name-1": { + # "size": 100, + # "tier": "maxiops", + # }, + # "some-disk-name-2": { + # "size": 100, + # "tier": "maxiops", + # } + } + } +} + +firewall_enabled = false +firewall_default_deny_in = false +firewall_default_deny_out = false + + +master_allowed_remote_ips = [ + { + "start_address" : "0.0.0.0" + "end_address" : "255.255.255.255" + } +] + +k8s_allowed_remote_ips = [ + { + "start_address" : "0.0.0.0" + "end_address" : "255.255.255.255" + } +] + +master_allowed_ports = [] +worker_allowed_ports = [] + +loadbalancer_enabled = false +loadbalancer_plan = "development" +loadbalancers = { + # "http" : { + # "port" : 80, + # "target_port" : 80, + # "backend_servers" : [ + # "worker-0", + # "worker-1", + # "worker-2" + # ] + # } +} + +server_groups = { + # "control-plane" = { + # servers = [ + # "master-0" + # ] + # anti_affinity = true + # }, + # "workers" = { + # servers = [ + # "worker-0", + # "worker-1", + # "worker-2" + # ] + # anti_affinity = true + # } +} \ No newline at end of file diff --git a/kubespray/contrib/terraform/upcloud/sample-inventory/group_vars b/kubespray/contrib/terraform/upcloud/sample-inventory/group_vars new file mode 120000 index 0000000..0d51062 --- /dev/null +++ b/kubespray/contrib/terraform/upcloud/sample-inventory/group_vars @@ -0,0 +1 @@ +../../../../inventory/sample/group_vars/ \ No newline at end of file diff --git a/kubespray/contrib/terraform/upcloud/templates/inventory.tpl b/kubespray/contrib/terraform/upcloud/templates/inventory.tpl new file mode 100644 index 0000000..28ff28a --- /dev/null +++ b/kubespray/contrib/terraform/upcloud/templates/inventory.tpl @@ -0,0 +1,17 @@ + +[all] +${connection_strings_master} +${connection_strings_worker} + +[kube_control_plane] +${list_master} + +[etcd] +${list_master} + +[kube_node] +${list_worker} + +[k8s_cluster:children] +kube_control_plane +kube_node diff --git a/kubespray/contrib/terraform/upcloud/variables.tf b/kubespray/contrib/terraform/upcloud/variables.tf new file mode 100644 index 0000000..3b2c558 --- /dev/null +++ b/kubespray/contrib/terraform/upcloud/variables.tf @@ -0,0 +1,144 @@ +variable "prefix" { + type = string + default = "kubespray" + + description = "Prefix that is used to distinguish these resources from others" +} + +variable "zone" { + description = "The zone where to run the cluster" +} + +variable "template_name" { + description = "Block describing the preconfigured operating system" +} + +variable "username" { + description = "The username to use for the nodes" + default = "ubuntu" +} + +variable "private_network_cidr" { + description = "CIDR to use for the private network" + default = "172.16.0.0/24" +} + +variable "machines" { + description = "Cluster machines" + + type = map(object({ + node_type = string + plan = string + cpu = string + mem = string + disk_size = number + additional_disks = map(object({ + size = number + tier = string + })) + })) +} + +variable "ssh_public_keys" { + description = "List of public SSH keys which are injected into the VMs." + type = list(string) +} + +variable "inventory_file" { + description = "Where to store the generated inventory file" +} + +variable "UPCLOUD_USERNAME" { + description = "UpCloud username with API access" +} + +variable "UPCLOUD_PASSWORD" { + description = "Password for UpCloud API user" +} + +variable "firewall_enabled" { + description = "Enable firewall rules" + default = false +} + +variable "master_allowed_remote_ips" { + description = "List of IP start/end addresses allowed to access API of masters" + type = list(object({ + start_address = string + end_address = string + })) + default = [] +} + +variable "k8s_allowed_remote_ips" { + description = "List of IP start/end addresses allowed to SSH to hosts" + type = list(object({ + start_address = string + end_address = string + })) + default = [] +} + +variable "master_allowed_ports" { + description = "List of ports to allow on masters" + type = list(object({ + protocol = string + port_range_min = number + port_range_max = number + start_address = string + end_address = string + })) +} + +variable "worker_allowed_ports" { + description = "List of ports to allow on workers" + type = list(object({ + protocol = string + port_range_min = number + port_range_max = number + start_address = string + end_address = string + })) +} + +variable "firewall_default_deny_in" { + description = "Add firewall policies that deny all inbound traffic by default" + default = false +} + +variable "firewall_default_deny_out" { + description = "Add firewall policies that deny all outbound traffic by default" + default = false +} + +variable "loadbalancer_enabled" { + description = "Enable load balancer" + default = false +} + +variable "loadbalancer_plan" { + description = "Load balancer plan (development/production-small)" + default = "development" +} + +variable "loadbalancers" { + description = "Load balancers" + + type = map(object({ + port = number + target_port = number + backend_servers = list(string) + })) + default = {} +} + +variable "server_groups" { + description = "Server groups" + + type = map(object({ + anti_affinity = bool + servers = list(string) + })) + + default = {} +} diff --git a/kubespray/contrib/terraform/upcloud/versions.tf b/kubespray/contrib/terraform/upcloud/versions.tf new file mode 100644 index 0000000..9950747 --- /dev/null +++ b/kubespray/contrib/terraform/upcloud/versions.tf @@ -0,0 +1,10 @@ + +terraform { + required_providers { + upcloud = { + source = "UpCloudLtd/upcloud" + version = "~>2.7.1" + } + } + required_version = ">= 0.13" +} diff --git a/kubespray/contrib/terraform/vsphere/README.md b/kubespray/contrib/terraform/vsphere/README.md new file mode 100644 index 0000000..7aa50d8 --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/README.md @@ -0,0 +1,128 @@ +# Kubernetes on vSphere with Terraform + +Provision a Kubernetes cluster on [vSphere](https://www.vmware.com/products/vsphere.html) using Terraform and Kubespray. + +## Overview + +The setup looks like following. + +```text + Kubernetes cluster ++-----------------------+ +| +--------------+ | +| | +--------------+ | +| | | | | +| | | Master/etcd | | +| | | node(s) | | +| +-+ | | +| +--------------+ | +| ^ | +| | | +| v | +| +--------------+ | +| | +--------------+ | +| | | | | +| | | Worker | | +| | | node(s) | | +| +-+ | | +| +--------------+ | ++-----------------------+ +``` + +## Warning + +This setup assumes that the DHCP is disabled in the vSphere cluster and IP addresses have to be provided in the configuration file. + +## Requirements + +* Terraform 0.13.0 or newer (0.12 also works if you modify the provider block to include version and remove all `versions.tf` files) + +## Quickstart + +NOTE: *Assumes you are at the root of the kubespray repo* + +Copy the sample inventory for your cluster and copy the default terraform variables. + +```bash +CLUSTER=my-vsphere-cluster +cp -r inventory/sample inventory/$CLUSTER +cp contrib/terraform/vsphere/default.tfvars inventory/$CLUSTER/ +cd inventory/$CLUSTER +``` + +Edit `default.tfvars` to match your setup. You MUST set values specific for you network and vSphere cluster. + +```bash +# Ensure $EDITOR points to your favorite editor, e.g., vim, emacs, VS Code, etc. +$EDITOR default.tfvars +``` + +For authentication in your vSphere cluster you can use the environment variables. + +```bash +export TF_VAR_vsphere_user=username +export TF_VAR_vsphere_password=password +``` + +Run Terraform to create the infrastructure. + +```bash +terraform init ../../contrib/terraform/vsphere +terraform apply \ + -var-file default.tfvars \ + -state=tfstate-$CLUSTER.tfstate \ + ../../contrib/terraform/vsphere +``` + +You should now have a inventory file named `inventory.ini` that you can use with kubespray. +You can now copy your inventory file and use it with kubespray to set up a cluster. +You can type `terraform output` to find out the IP addresses of the nodes. + +It is a good idea to check that you have basic SSH connectivity to the nodes. You can do that by: + +```bash +ansible -i inventory.ini -m ping all +``` + +Example to use this with the default sample inventory: + +```bash +ansible-playbook -i inventory.ini ../../cluster.yml -b -v +``` + +## Variables + +### Required + +* `machines`: Machines to provision. Key of this object will be used as the name of the machine + * `node_type`: The role of this node *(master|worker)* + * `ip`: The IP address of the machine + * `netmask`: The netmask to use (to be used on the right hand side in CIDR notation, e.g., `24`) +* `network`: The name of the network to attach the machines to +* `gateway`: The IP address of the network gateway +* `vsphere_datacenter`: The identifier of vSphere data center +* `vsphere_compute_cluster`: The identifier of vSphere compute cluster +* `vsphere_datastore`: The identifier of vSphere data store +* `vsphere_server`: This is the vCenter server name or address for vSphere API operations. +* `ssh_public_keys`: List of public SSH keys to install on all machines +* `template_name`: The name of a base image (the OVF template be defined in vSphere beforehand) + +### Optional + +* `folder`: Name of the folder to put all machines in (default: `""`) +* `prefix`: Prefix to use for all resources, required to be unique for all clusters in the same project (default: `"k8s"`) +* `inventory_file`: Name of the generated inventory file for Kubespray to use in the Ansible step (default: `inventory.ini`) +* `dns_primary`: The IP address of primary DNS server (default: `8.8.4.4`) +* `dns_secondary`: The IP address of secondary DNS server (default: `8.8.8.8`) +* `firmware`: Firmware to use (default: `bios`) +* `hardware_version`: The version of the hardware (default: `15`) +* `master_cores`: The number of CPU cores for the master nodes (default: 4) +* `master_memory`: The amount of RAM for the master nodes in MB (default: 4096) +* `master_disk_size`: The amount of disk space for the master nodes in GB (default: 20) +* `worker_cores`: The number of CPU cores for the worker nodes (default: 16) +* `worker_memory`: The amount of RAM for the worker nodes in MB (default: 8192) +* `worker_disk_size`: The amount of disk space for the worker nodes in GB (default: 100) +* `vapp`: Boolean to set the template type to vapp. (Default: false) +* `interface_name`: Name of the interface to configure. (Default: ens192) + +An example variables file can be found `default.tfvars` diff --git a/kubespray/contrib/terraform/vsphere/default.tfvars b/kubespray/contrib/terraform/vsphere/default.tfvars new file mode 100644 index 0000000..fa16936 --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/default.tfvars @@ -0,0 +1,38 @@ +prefix = "k8s" + +inventory_file = "inventory.ini" + +network = "VM Network" + +machines = { + "master-0" : { + "node_type" : "master", + "ip" : "i-did-not-read-the-docs", # e.g. 192.168.0.10 + "netmask" : "24" + }, + "worker-0" : { + "node_type" : "worker", + "ip" : "i-did-not-read-the-docs", # e.g. 192.168.0.20 + "netmask" : "24" + }, + "worker-1" : { + "node_type" : "worker", + "ip" : "i-did-not-read-the-docs", # e.g. 192.168.0.21 + "netmask" : "24" + } +} + +gateway = "i-did-not-read-the-docs" # e.g. 192.168.0.1 + +ssh_public_keys = [ + # Put your public SSH key here + "ssh-rsa I-did-not-read-the-docs", + "ssh-rsa I-did-not-read-the-docs 2", +] + +vsphere_datacenter = "i-did-not-read-the-docs" +vsphere_compute_cluster = "i-did-not-read-the-docs" # e.g. Cluster +vsphere_datastore = "i-did-not-read-the-docs" # e.g. ssd-000000 +vsphere_server = "i-did-not-read-the-docs" # e.g. vsphere.server.com + +template_name = "i-did-not-read-the-docs" # e.g. ubuntu-bionic-18.04-cloudimg diff --git a/kubespray/contrib/terraform/vsphere/main.tf b/kubespray/contrib/terraform/vsphere/main.tf new file mode 100644 index 0000000..fb2d8c8 --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/main.tf @@ -0,0 +1,100 @@ +provider "vsphere" { + # Username and password set through env vars VSPHERE_USER and VSPHERE_PASSWORD + user = var.vsphere_user + password = var.vsphere_password + + vsphere_server = var.vsphere_server + + # If you have a self-signed cert + allow_unverified_ssl = true +} + +data "vsphere_datacenter" "dc" { + name = var.vsphere_datacenter +} + +data "vsphere_datastore" "datastore" { + name = var.vsphere_datastore + datacenter_id = data.vsphere_datacenter.dc.id +} + +data "vsphere_network" "network" { + name = var.network + datacenter_id = data.vsphere_datacenter.dc.id +} + +data "vsphere_virtual_machine" "template" { + name = var.template_name + datacenter_id = data.vsphere_datacenter.dc.id +} + +data "vsphere_compute_cluster" "compute_cluster" { + name = var.vsphere_compute_cluster + datacenter_id = data.vsphere_datacenter.dc.id +} + +resource "vsphere_resource_pool" "pool" { + name = "${var.prefix}-cluster-pool" + parent_resource_pool_id = data.vsphere_compute_cluster.compute_cluster.resource_pool_id +} + +module "kubernetes" { + source = "./modules/kubernetes-cluster" + + prefix = var.prefix + + machines = var.machines + + ## Master ## + master_cores = var.master_cores + master_memory = var.master_memory + master_disk_size = var.master_disk_size + + ## Worker ## + worker_cores = var.worker_cores + worker_memory = var.worker_memory + worker_disk_size = var.worker_disk_size + + ## Global ## + + gateway = var.gateway + dns_primary = var.dns_primary + dns_secondary = var.dns_secondary + + pool_id = vsphere_resource_pool.pool.id + datastore_id = data.vsphere_datastore.datastore.id + + folder = var.folder + guest_id = data.vsphere_virtual_machine.template.guest_id + scsi_type = data.vsphere_virtual_machine.template.scsi_type + network_id = data.vsphere_network.network.id + adapter_type = data.vsphere_virtual_machine.template.network_interface_types[0] + interface_name = var.interface_name + firmware = var.firmware + hardware_version = var.hardware_version + disk_thin_provisioned = data.vsphere_virtual_machine.template.disks.0.thin_provisioned + + template_id = data.vsphere_virtual_machine.template.id + vapp = var.vapp + + ssh_public_keys = var.ssh_public_keys +} + +# +# Generate ansible inventory +# + +resource "local_file" "inventory" { + content = templatefile("${path.module}/templates/inventory.tpl", { + connection_strings_master = join("\n", formatlist("%s ansible_user=ubuntu ansible_host=%s etcd_member_name=etcd%d", + keys(module.kubernetes.master_ip), + values(module.kubernetes.master_ip), + range(1, length(module.kubernetes.master_ip) + 1))), + connection_strings_worker = join("\n", formatlist("%s ansible_user=ubuntu ansible_host=%s", + keys(module.kubernetes.worker_ip), + values(module.kubernetes.worker_ip))), + list_master = join("\n", formatlist("%s", keys(module.kubernetes.master_ip))), + list_worker = join("\n", formatlist("%s", keys(module.kubernetes.worker_ip))) + }) + filename = var.inventory_file +} diff --git a/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/main.tf b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/main.tf new file mode 100644 index 0000000..a44c2cf --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/main.tf @@ -0,0 +1,149 @@ +resource "vsphere_virtual_machine" "worker" { + for_each = { + for name, machine in var.machines : + name => machine + if machine.node_type == "worker" + } + + name = "${var.prefix}-${each.key}" + + resource_pool_id = var.pool_id + datastore_id = var.datastore_id + + num_cpus = var.worker_cores + memory = var.worker_memory + memory_reservation = var.worker_memory + guest_id = var.guest_id + enable_disk_uuid = "true" # needed for CSI provider + scsi_type = var.scsi_type + folder = var.folder + firmware = var.firmware + hardware_version = var.hardware_version + + wait_for_guest_net_routable = false + wait_for_guest_net_timeout = 0 + + network_interface { + network_id = var.network_id + adapter_type = var.adapter_type + } + + disk { + label = "disk0" + size = var.worker_disk_size + thin_provisioned = var.disk_thin_provisioned + } + + lifecycle { + ignore_changes = [disk] + } + + clone { + template_uuid = var.template_id + } + + cdrom { + client_device = true + } + + dynamic "vapp" { + for_each = var.vapp ? [1] : [] + + content { + properties = { + "user-data" = base64encode(templatefile("${path.module}/templates/vapp-cloud-init.tpl", { ssh_public_keys = var.ssh_public_keys })) + } + } + } + + extra_config = { + "isolation.tools.copy.disable" = "FALSE" + "isolation.tools.paste.disable" = "FALSE" + "isolation.tools.setGUIOptions.enable" = "TRUE" + "guestinfo.userdata" = base64encode(templatefile("${path.module}/templates/cloud-init.tpl", { ssh_public_keys = var.ssh_public_keys })) + "guestinfo.userdata.encoding" = "base64" + "guestinfo.metadata" = base64encode(templatefile("${path.module}/templates/metadata.tpl", { hostname = "${var.prefix}-${each.key}", + interface_name = var.interface_name + ip = each.value.ip, + netmask = each.value.netmask, + gw = var.gateway, + dns = var.dns_primary, + ssh_public_keys = var.ssh_public_keys })) + "guestinfo.metadata.encoding" = "base64" + } +} + +resource "vsphere_virtual_machine" "master" { + for_each = { + for name, machine in var.machines : + name => machine + if machine.node_type == "master" + } + + name = "${var.prefix}-${each.key}" + + resource_pool_id = var.pool_id + datastore_id = var.datastore_id + + num_cpus = var.master_cores + memory = var.master_memory + memory_reservation = var.master_memory + guest_id = var.guest_id + enable_disk_uuid = "true" # needed for CSI provider + scsi_type = var.scsi_type + folder = var.folder + firmware = var.firmware + hardware_version = var.hardware_version + + wait_for_guest_net_routable = false + wait_for_guest_net_timeout = 0 + + network_interface { + network_id = var.network_id + adapter_type = var.adapter_type + } + + disk { + label = "disk0" + size = var.master_disk_size + thin_provisioned = var.disk_thin_provisioned + } + + lifecycle { + ignore_changes = [disk] + } + + clone { + template_uuid = var.template_id + } + + cdrom { + client_device = true + } + + dynamic "vapp" { + for_each = var.vapp ? [1] : [] + + content { + properties = { + "user-data" = base64encode(templatefile("${path.module}/templates/vapp-cloud-init.tpl", { ssh_public_keys = var.ssh_public_keys })) + } + } + } + + extra_config = { + "isolation.tools.copy.disable" = "FALSE" + "isolation.tools.paste.disable" = "FALSE" + "isolation.tools.setGUIOptions.enable" = "TRUE" + "guestinfo.userdata" = base64encode(templatefile("${path.module}/templates/cloud-init.tpl", { ssh_public_keys = var.ssh_public_keys })) + "guestinfo.userdata.encoding" = "base64" + "guestinfo.metadata" = base64encode(templatefile("${path.module}/templates/metadata.tpl", { hostname = "${var.prefix}-${each.key}", + interface_name = var.interface_name + ip = each.value.ip, + netmask = each.value.netmask, + gw = var.gateway, + dns = var.dns_primary, + ssh_public_keys = var.ssh_public_keys })) + "guestinfo.metadata.encoding" = "base64" + } +} diff --git a/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/output.tf b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/output.tf new file mode 100644 index 0000000..93752ab --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/output.tf @@ -0,0 +1,15 @@ +output "master_ip" { + value = { + for name, machine in var.machines : + "${var.prefix}-${name}" => machine.ip + if machine.node_type == "master" + } +} + +output "worker_ip" { + value = { + for name, machine in var.machines : + "${var.prefix}-${name}" => machine.ip + if machine.node_type == "worker" + } +} diff --git a/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/templates/cloud-init.tpl b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/templates/cloud-init.tpl new file mode 100644 index 0000000..5f809af --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/templates/cloud-init.tpl @@ -0,0 +1,6 @@ +#cloud-config + +ssh_authorized_keys: +%{ for ssh_public_key in ssh_public_keys ~} + - ${ssh_public_key} +%{ endfor ~} diff --git a/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/templates/metadata.tpl b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/templates/metadata.tpl new file mode 100644 index 0000000..1553f08 --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/templates/metadata.tpl @@ -0,0 +1,14 @@ +instance-id: ${hostname} +local-hostname: ${hostname} +network: + version: 2 + ethernets: + ${interface_name}: + match: + name: ${interface_name} + dhcp4: false + addresses: + - ${ip}/${netmask} + gateway4: ${gw} + nameservers: + addresses: [${dns}] diff --git a/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/templates/vapp-cloud-init.tpl b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/templates/vapp-cloud-init.tpl new file mode 100644 index 0000000..07d0778 --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/templates/vapp-cloud-init.tpl @@ -0,0 +1,24 @@ +#cloud-config + +ssh_authorized_keys: +%{ for ssh_public_key in ssh_public_keys ~} + - ${ssh_public_key} +%{ endfor ~} + +write_files: + - path: /etc/netplan/10-user-network.yaml + content: |. + network: + version: 2 + ethernets: + ${interface_name}: + dhcp4: false #true to use dhcp + addresses: + - ${ip}/${netmask} + gateway4: ${gw} # Set gw here + nameservers: + addresses: + - ${dns} # Set DNS ip address here + +runcmd: + - netplan apply diff --git a/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/variables.tf b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/variables.tf new file mode 100644 index 0000000..cb99142 --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/variables.tf @@ -0,0 +1,43 @@ +## Global ## +variable "prefix" {} + +variable "machines" { + description = "Cluster machines" + type = map(object({ + node_type = string + ip = string + netmask = string + })) +} + +variable "gateway" {} +variable "dns_primary" {} +variable "dns_secondary" {} +variable "pool_id" {} +variable "datastore_id" {} +variable "guest_id" {} +variable "scsi_type" {} +variable "network_id" {} +variable "interface_name" {} +variable "adapter_type" {} +variable "disk_thin_provisioned" {} +variable "template_id" {} +variable "vapp" { + type = bool +} +variable "firmware" {} +variable "folder" {} +variable "ssh_public_keys" { + type = list(string) +} +variable "hardware_version" {} + +## Master ## +variable "master_cores" {} +variable "master_memory" {} +variable "master_disk_size" {} + +## Worker ## +variable "worker_cores" {} +variable "worker_memory" {} +variable "worker_disk_size" {} diff --git a/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/versions.tf b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/versions.tf new file mode 100644 index 0000000..8c622fd --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/modules/kubernetes-cluster/versions.tf @@ -0,0 +1,9 @@ +terraform { + required_providers { + vsphere = { + source = "hashicorp/vsphere" + version = ">= 1.24.3" + } + } + required_version = ">= 0.13" +} diff --git a/kubespray/contrib/terraform/vsphere/output.tf b/kubespray/contrib/terraform/vsphere/output.tf new file mode 100644 index 0000000..a4338d9 --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/output.tf @@ -0,0 +1,31 @@ +output "master_ip_addresses" { + value = module.kubernetes.master_ip +} + +output "worker_ip_addresses" { + value = module.kubernetes.worker_ip +} + +output "vsphere_datacenter" { + value = var.vsphere_datacenter +} + +output "vsphere_server" { + value = var.vsphere_server +} + +output "vsphere_datastore" { + value = var.vsphere_datastore +} + +output "vsphere_network" { + value = var.network +} + +output "vsphere_folder" { + value = var.folder +} + +output "vsphere_pool" { + value = "${terraform.workspace}-cluster-pool" +} diff --git a/kubespray/contrib/terraform/vsphere/sample-inventory/cluster.tfvars b/kubespray/contrib/terraform/vsphere/sample-inventory/cluster.tfvars new file mode 100644 index 0000000..dfa0a3d --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/sample-inventory/cluster.tfvars @@ -0,0 +1,33 @@ +prefix = "default" + +inventory_file = "inventory.ini" + +machines = { + "master-0" : { + "node_type" : "master", + "ip" : "i-did-not-read-the-docs" # e.g. 192.168.0.2/24 + }, + "worker-0" : { + "node_type" : "worker", + "ip" : "i-did-not-read-the-docs" # e.g. 192.168.0.2/24 + }, + "worker-1" : { + "node_type" : "worker", + "ip" : "i-did-not-read-the-docs" # e.g. 192.168.0.2/24 + } +} + +gateway = "i-did-not-read-the-docs" # e.g. 192.168.0.2 + +ssh_public_keys = [ + # Put your public SSH key here + "ssh-rsa I-did-not-read-the-docs", + "ssh-rsa I-did-not-read-the-docs 2", +] + +vsphere_datacenter = "i-did-not-read-the-docs" +vsphere_compute_cluster = "i-did-not-read-the-docs" # e.g. Cluster +vsphere_datastore = "i-did-not-read-the-docs" # e.g. ssd-000000 +vsphere_server = "i-did-not-read-the-docs" # e.g. vsphere.server.com + +template_name = "i-did-not-read-the-docs" # e.g. ubuntu-bionic-18.04-cloudimg diff --git a/kubespray/contrib/terraform/vsphere/sample-inventory/group_vars b/kubespray/contrib/terraform/vsphere/sample-inventory/group_vars new file mode 120000 index 0000000..3735958 --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/sample-inventory/group_vars @@ -0,0 +1 @@ +../../../../inventory/sample/group_vars \ No newline at end of file diff --git a/kubespray/contrib/terraform/vsphere/templates/inventory.tpl b/kubespray/contrib/terraform/vsphere/templates/inventory.tpl new file mode 100644 index 0000000..28ff28a --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/templates/inventory.tpl @@ -0,0 +1,17 @@ + +[all] +${connection_strings_master} +${connection_strings_worker} + +[kube_control_plane] +${list_master} + +[etcd] +${list_master} + +[kube_node] +${list_worker} + +[k8s_cluster:children] +kube_control_plane +kube_node diff --git a/kubespray/contrib/terraform/vsphere/variables.tf b/kubespray/contrib/terraform/vsphere/variables.tf new file mode 100644 index 0000000..03f9007 --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/variables.tf @@ -0,0 +1,101 @@ +## Global ## + +# Required variables + +variable "machines" { + description = "Cluster machines" + type = map(object({ + node_type = string + ip = string + netmask = string + })) +} + +variable "network" {} + +variable "gateway" {} + +variable "vsphere_datacenter" {} + +variable "vsphere_compute_cluster" {} + +variable "vsphere_datastore" {} + +variable "vsphere_user" {} + +variable "vsphere_password" { + sensitive = true +} + +variable "vsphere_server" {} + +variable "ssh_public_keys" { + description = "List of public SSH keys which are injected into the VMs." + type = list(string) +} + +variable "template_name" {} + +# Optional variables (ones where reasonable defaults exist) +variable "vapp" { + default = false +} + +variable "interface_name" { + default = "ens192" +} + +variable "folder" { + default = "" +} + +variable "prefix" { + default = "k8s" +} + +variable "inventory_file" { + default = "inventory.ini" +} + +variable "dns_primary" { + default = "8.8.4.4" +} + +variable "dns_secondary" { + default = "8.8.8.8" +} + +variable "firmware" { + default = "bios" +} + +variable "hardware_version" { + default = "15" +} + +## Master ## + +variable "master_cores" { + default = 4 +} + +variable "master_memory" { + default = 4096 +} + +variable "master_disk_size" { + default = "20" +} + +## Worker ## + +variable "worker_cores" { + default = 16 +} + +variable "worker_memory" { + default = 8192 +} +variable "worker_disk_size" { + default = "100" +} diff --git a/kubespray/contrib/terraform/vsphere/versions.tf b/kubespray/contrib/terraform/vsphere/versions.tf new file mode 100644 index 0000000..8c622fd --- /dev/null +++ b/kubespray/contrib/terraform/vsphere/versions.tf @@ -0,0 +1,9 @@ +terraform { + required_providers { + vsphere = { + source = "hashicorp/vsphere" + version = ">= 1.24.3" + } + } + required_version = ">= 0.13" +} diff --git a/kubespray/docs/_sidebar.md b/kubespray/docs/_sidebar.md new file mode 100644 index 0000000..cbc352b --- /dev/null +++ b/kubespray/docs/_sidebar.md @@ -0,0 +1,67 @@ +* [Readme](/) +* [Comparisons](/docs/comparisons.md) +* [Getting started](/docs/getting-started.md) +* [Ansible](docs/ansible.md) +* [Variables](/docs/vars.md) +* Operations + * [Integration](docs/integration.md) + * [Upgrades](/docs/upgrades.md) + * [HA Mode](docs/ha-mode.md) + * [Adding/replacing a node](docs/nodes.md) + * [Large deployments](docs/large-deployments.md) + * [Air-Gap Installation](docs/offline-environment.md) +* CNI + * [Calico](docs/calico.md) + * [Flannel](docs/flannel.md) + * [Cilium](docs/cilium.md) + * [Kube Router](docs/kube-router.md) + * [Kube OVN](docs/kube-ovn.md) + * [Weave](docs/weave.md) + * [Multus](docs/multus.md) +* Ingress + * [kube-vip](docs/kube-vip.md) + * [ALB Ingress](docs/ingress_controller/alb_ingress_controller.md) + * [MetalLB](docs/metallb.md) + * [Nginx Ingress](docs/ingress_controller/ingress_nginx.md) +* [Cloud providers](docs/cloud.md) + * [AWS](docs/aws.md) + * [Azure](docs/azure.md) + * [OpenStack](/docs/openstack.md) + * [Equinix Metal](/docs/equinix-metal.md) + * [vSphere](/docs/vsphere.md) +* [Operating Systems](docs/bootstrap-os.md) + * [Debian](docs/debian.md) + * [Flatcar Container Linux](docs/flatcar.md) + * [Fedora CoreOS](docs/fcos.md) + * [OpenSUSE](docs/opensuse.md) + * [RedHat Enterprise Linux](docs/rhel.md) + * [CentOS/OracleLinux/AlmaLinux/Rocky Linux](docs/centos.md) + * [Kylin Linux Advanced Server V10](docs/kylinlinux.md) + * [Amazon Linux 2](docs/amazonlinux.md) + * [UOS Linux](docs/uoslinux.md) + * [openEuler notes](docs/openeuler.md) +* CRI + * [Containerd](docs/containerd.md) + * [Docker](docs/docker.md) + * [CRI-O](docs/cri-o.md) + * [Kata Containers](docs/kata-containers.md) + * [gVisor](docs/gvisor.md) +* Advanced + * [Proxy](/docs/proxy.md) + * [Downloads](docs/downloads.md) + * [Netcheck](docs/netcheck.md) + * [Cert Manager](docs/cert_manager.md) + * [DNS Stack](docs/dns-stack.md) + * [Kubernetes reliability](docs/kubernetes-reliability.md) + * [Local Registry](docs/kubernetes-apps/registry.md) + * [NTP](docs/ntp.md) +* External Storage Provisioners + * [RBD Provisioner](docs/kubernetes-apps/rbd_provisioner.md) + * [CEPHFS Provisioner](docs/kubernetes-apps/cephfs_provisioner.md) + * [Local Volume Provisioner](docs/kubernetes-apps/local_volume_provisioner.md) +* Developers + * [Test cases](docs/test_cases.md) + * [Vagrant](docs/vagrant.md) + * [CI Matrix](docs/ci.md) + * [CI Setup](docs/ci-setup.md) +* [Roadmap](docs/roadmap.md) diff --git a/kubespray/docs/amazonlinux.md b/kubespray/docs/amazonlinux.md new file mode 100644 index 0000000..9609171 --- /dev/null +++ b/kubespray/docs/amazonlinux.md @@ -0,0 +1,15 @@ +# Amazon Linux 2 + +Amazon Linux is supported with docker,containerd and cri-o runtimes. + +**Note:** that Amazon Linux is not currently covered in kubespray CI and +support for it is currently considered experimental. + +Amazon Linux 2, while derived from the Redhat OS family, does not keep in +sync with RHEL upstream like CentOS/AlmaLinux/Oracle Linux. In order to use +Amazon Linux as the ansible host for your kubespray deployments you need to +manually install `python3` and deploy ansible and kubespray dependencies in +a python virtual environment or use the official kubespray containers. + +There are no special considerations for using Amazon Linux as the target OS +for Kubespray deployments. diff --git a/kubespray/docs/ansible.md b/kubespray/docs/ansible.md new file mode 100644 index 0000000..9a3110b --- /dev/null +++ b/kubespray/docs/ansible.md @@ -0,0 +1,307 @@ +# Ansible + +## Installing Ansible + +Kubespray supports multiple ansible versions and ships different `requirements.txt` files for them. +Depending on your available python version you may be limited in choosing which ansible version to use. + +It is recommended to deploy the ansible version used by kubespray into a python virtual environment. + +```ShellSession +VENVDIR=kubespray-venv +KUBESPRAYDIR=kubespray +python3 -m venv $VENVDIR +source $VENVDIR/bin/activate +cd $KUBESPRAYDIR +pip install -U -r requirements.txt +``` + +In case you have a similar message when installing the requirements: + +```ShellSession +ERROR: Could not find a version that satisfies the requirement ansible==7.6.0 (from -r requirements.txt (line 1)) (from versions: [...], 6.7.0) +ERROR: No matching distribution found for ansible==7.6.0 (from -r requirements.txt (line 1)) +``` + +It means that the version of Python you are running is not compatible with the version of Ansible that Kubespray supports. +If the latest version supported according to pip is 6.7.0 it means you are running Python 3.8 or lower while you need at least Python 3.9 (see the table below). + +### Ansible Python Compatibility + +Based on the table below and the available python version for your ansible host you should choose the appropriate ansible version to use with kubespray. + +| Ansible Version | Python Version | +|-----------------|----------------| +| 2.14 | 3.9-3.11 | + +## Inventory + +The inventory is composed of 3 groups: + +* **kube_node** : list of kubernetes nodes where the pods will run. +* **kube_control_plane** : list of servers where kubernetes control plane components (apiserver, scheduler, controller) will run. +* **etcd**: list of servers to compose the etcd server. You should have at least 3 servers for failover purpose. + +Note: do not modify the children of _k8s_cluster_, like putting +the _etcd_ group into the _k8s_cluster_, unless you are certain +to do that and you have it fully contained in the latter: + +```ShellSession +etcd ⊂ k8s_cluster => kube_node ∩ etcd = etcd +``` + +When _kube_node_ contains _etcd_, you define your etcd cluster to be as well schedulable for Kubernetes workloads. +If you want it a standalone, make sure those groups do not intersect. +If you want the server to act both as control-plane and node, the server must be defined +on both groups _kube_control_plane_ and _kube_node_. If you want a standalone and +unschedulable control plane, the server must be defined only in the _kube_control_plane_ and +not _kube_node_. + +There are also two special groups: + +* **calico_rr** : explained for [advanced Calico networking cases](/docs/calico.md) +* **bastion** : configure a bastion host if your nodes are not directly reachable + +Below is a complete inventory example: + +```ini +## Configure 'ip' variable to bind kubernetes services on a +## different ip than the default iface +node1 ansible_host=95.54.0.12 ip=10.3.0.1 +node2 ansible_host=95.54.0.13 ip=10.3.0.2 +node3 ansible_host=95.54.0.14 ip=10.3.0.3 +node4 ansible_host=95.54.0.15 ip=10.3.0.4 +node5 ansible_host=95.54.0.16 ip=10.3.0.5 +node6 ansible_host=95.54.0.17 ip=10.3.0.6 + +[kube_control_plane] +node1 +node2 + +[etcd] +node1 +node2 +node3 + +[kube_node] +node2 +node3 +node4 +node5 +node6 + +[k8s_cluster:children] +kube_node +kube_control_plane +``` + +## Group vars and overriding variables precedence + +The group variables to control main deployment options are located in the directory ``inventory/sample/group_vars``. +Optional variables are located in the `inventory/sample/group_vars/all.yml`. +Mandatory variables that are common for at least one role (or a node group) can be found in the +`inventory/sample/group_vars/k8s_cluster.yml`. +There are also role vars for docker, kubernetes preinstall and control plane roles. +According to the [ansible docs](https://docs.ansible.com/ansible/latest/playbooks_variables.html#variable-precedence-where-should-i-put-a-variable), +those cannot be overridden from the group vars. In order to override, one should use +the `-e` runtime flags (most simple way) or other layers described in the docs. + +Kubespray uses only a few layers to override things (or expect them to +be overridden for roles): + +| Layer | Comment | +|----------------------------------------|------------------------------------------------------------------------------| +| **role defaults** | provides best UX to override things for Kubespray deployments | +| inventory vars | Unused | +| **inventory group_vars** | Expects users to use ``all.yml``,``k8s_cluster.yml`` etc. to override things | +| inventory host_vars | Unused | +| playbook group_vars | Unused | +| playbook host_vars | Unused | +| **host facts** | Kubespray overrides for internal roles' logic, like state flags | +| play vars | Unused | +| play vars_prompt | Unused | +| play vars_files | Unused | +| registered vars | Unused | +| set_facts | Kubespray overrides those, for some places | +| **role and include vars** | Provides bad UX to override things! Use extra vars to enforce | +| block vars (only for tasks in block) | Kubespray overrides for internal roles' logic | +| task vars (only for the task) | Unused for roles, but only for helper scripts | +| **extra vars** (always win precedence) | override with ``ansible-playbook -e @foo.yml`` | + +## Ansible tags + +The following tags are defined in playbooks: + +| Tag name | Used for | +|--------------------------------|-------------------------------------------------------| +| annotate | Create kube-router annotation | +| apps | K8s apps definitions | +| asserts | Check tasks for download role | +| aws-ebs-csi-driver | Configuring csi driver: aws-ebs | +| azure-csi-driver | Configuring csi driver: azure | +| bastion | Setup ssh config for bastion | +| bootstrap-os | Anything related to host OS configuration | +| calico | Network plugin Calico | +| calico_rr | Configuring Calico route reflector | +| cephfs-provisioner | Configuring CephFS | +| cert-manager | Configuring certificate manager for K8s | +| cilium | Network plugin Cilium | +| cinder-csi-driver | Configuring csi driver: cinder | +| client | Kubernetes clients role | +| cloud-provider | Cloud-provider related tasks | +| cluster-roles | Configuring cluster wide application (psp ...) | +| cni | CNI plugins for Network Plugins | +| containerd | Configuring containerd engine runtime for hosts | +| container_engine_accelerator | Enable nvidia accelerator for runtimes | +| container-engine | Configuring container engines | +| container-runtimes | Configuring container runtimes | +| coredns | Configuring coredns deployment | +| crio | Configuring crio container engine for hosts | +| crun | Configuring crun runtime | +| csi-driver | Configuring csi driver | +| dashboard | Installing and configuring the Kubernetes Dashboard | +| dns | Remove dns entries when resetting | +| docker | Configuring docker engine runtime for hosts | +| download | Fetching container images to a delegate host | +| etcd | Configuring etcd cluster | +| etcd-secrets | Configuring etcd certs/keys | +| etchosts | Configuring /etc/hosts entries for hosts | +| external-cloud-controller | Configure cloud controllers | +| external-openstack | Cloud controller : openstack | +| external-provisioner | Configure external provisioners | +| external-vsphere | Cloud controller : vsphere | +| facts | Gathering facts and misc check results | +| files | Remove files when resetting | +| flannel | Network plugin flannel | +| gce | Cloud-provider GCP | +| gcp-pd-csi-driver | Configuring csi driver: gcp-pd | +| gvisor | Configuring gvisor runtime | +| helm | Installing and configuring Helm | +| ingress-controller | Configure ingress controllers | +| ingress_alb | AWS ALB Ingress Controller | +| init | Windows kubernetes init nodes | +| iptables | Flush and clear iptable when resetting | +| k8s-pre-upgrade | Upgrading K8s cluster | +| k8s-secrets | Configuring K8s certs/keys | +| k8s-gen-tokens | Configuring K8s tokens | +| kata-containers | Configuring kata-containers runtime | +| krew | Install and manage krew | +| kubeadm | Roles linked to kubeadm tasks | +| kube-apiserver | Configuring static pod kube-apiserver | +| kube-controller-manager | Configuring static pod kube-controller-manager | +| kube-vip | Installing and configuring kube-vip | +| kubectl | Installing kubectl and bash completion | +| kubelet | Configuring kubelet service | +| kube-ovn | Network plugin kube-ovn | +| kube-router | Network plugin kube-router | +| kube-proxy | Configuring static pod kube-proxy | +| localhost | Special steps for the localhost (ansible runner) | +| local-path-provisioner | Configure External provisioner: local-path | +| local-volume-provisioner | Configure External provisioner: local-volume | +| macvlan | Network plugin macvlan | +| master | Configuring K8s master node role | +| metallb | Installing and configuring metallb | +| metrics_server | Configuring metrics_server | +| netchecker | Installing netchecker K8s app | +| network | Configuring networking plugins for K8s | +| mounts | Umount kubelet dirs when reseting | +| multus | Network plugin multus | +| nginx | Configuring LB for kube-apiserver instances | +| node | Configuring K8s minion (compute) node role | +| nodelocaldns | Configuring nodelocaldns daemonset | +| node-label | Tasks linked to labeling of nodes | +| node-webhook | Tasks linked to webhook (grating access to resources) | +| nvidia_gpu | Enable nvidia accelerator for runtimes | +| oci | Cloud provider: oci | +| persistent_volumes | Configure csi volumes | +| persistent_volumes_aws_ebs_csi | Configuring csi driver: aws-ebs | +| persistent_volumes_cinder_csi | Configuring csi driver: cinder | +| persistent_volumes_gcp_pd_csi | Configuring csi driver: gcp-pd | +| persistent_volumes_openstack | Configuring csi driver: openstack | +| policy-controller | Configuring Calico policy controller | +| post-remove | Tasks running post-remove operation | +| post-upgrade | Tasks running post-upgrade operation | +| pre-remove | Tasks running pre-remove operation | +| pre-upgrade | Tasks running pre-upgrade operation | +| preinstall | Preliminary configuration steps | +| registry | Configuring local docker registry | +| reset | Tasks running doing the node reset | +| resolvconf | Configuring /etc/resolv.conf for hosts/apps | +| rbd-provisioner | Configure External provisioner: rdb | +| services | Remove services (etcd, kubelet etc...) when resetting | +| snapshot | Enabling csi snapshot | +| snapshot-controller | Configuring csi snapshot controller | +| upgrade | Upgrading, f.e. container images/binaries | +| upload | Distributing images/binaries across hosts | +| vsphere-csi-driver | Configuring csi driver: vsphere | +| weave | Network plugin Weave | +| win_nodes | Running windows specific tasks | +| youki | Configuring youki runtime | + +Note: Use the ``bash scripts/gen_tags.sh`` command to generate a list of all +tags found in the codebase. New tags will be listed with the empty "Used for" +field. + +## Example commands + +Example command to filter and apply only DNS configuration tasks and skip +everything else related to host OS configuration and downloading images of containers: + +```ShellSession +ansible-playbook -i inventory/sample/hosts.ini cluster.yml --tags preinstall,facts --skip-tags=download,bootstrap-os +``` + +And this play only removes the K8s cluster DNS resolver IP from hosts' /etc/resolv.conf files: + +```ShellSession +ansible-playbook -i inventory/sample/hosts.ini -e dns_mode='none' cluster.yml --tags resolvconf +``` + +And this prepares all container images locally (at the ansible runner node) without installing +or upgrading related stuff or trying to upload container to K8s cluster nodes: + +```ShellSession +ansible-playbook -i inventory/sample/hosts.ini cluster.yml \ + -e download_run_once=true -e download_localhost=true \ + --tags download --skip-tags upload,upgrade +``` + +Note: use `--tags` and `--skip-tags` wise and only if you're 100% sure what you're doing. + +## Bastion host + +If you prefer to not make your nodes publicly accessible (nodes with private IPs only), +you can use a so-called _bastion_ host to connect to your nodes. To specify and use a bastion, +simply add a line to your inventory, where you have to replace x.x.x.x with the public IP of the +bastion host. + +```ShellSession +[bastion] +bastion ansible_host=x.x.x.x +``` + +For more information about Ansible and bastion hosts, read +[Running Ansible Through an SSH Bastion Host](https://blog.scottlowe.org/2015/12/24/running-ansible-through-ssh-bastion-host/) + +## Mitogen + +Mitogen support is deprecated, please see [mitogen related docs](/docs/mitogen.md) for usage and reasons for deprecation. + +## Beyond ansible 2.9 + +Ansible project has decided, in order to ease their maintenance burden, to split between +two projects which are now joined under the Ansible umbrella. + +Ansible-base (2.10.x branch) will contain just the ansible language implementation while +ansible modules that were previously bundled into a single repository will be part of the +ansible 3.x package. Please see [this blog post](https://blog.while-true-do.io/ansible-release-3-0-0/) +that explains in detail the need and the evolution plan. + +**Note:** this change means that ansible virtual envs cannot be upgraded with `pip install -U`. +You first need to uninstall your old ansible (pre 2.10) version and install the new one. + +```ShellSession +pip uninstall ansible ansible-base ansible-core +cd kubespray/ +pip install -U . +``` diff --git a/kubespray/docs/ansible_collection.md b/kubespray/docs/ansible_collection.md new file mode 100644 index 0000000..6b559cc --- /dev/null +++ b/kubespray/docs/ansible_collection.md @@ -0,0 +1,38 @@ +# Ansible collection + +Kubespray can be installed as an [Ansible collection](https://docs.ansible.com/ansible/latest/user_guide/collections_using.html). + +## Requirements + +- An inventory file with the appropriate host groups. See the [README](../README.md#usage). +- A `group_vars` directory. These group variables **need** to match the appropriate variable names under `inventory/local/group_vars`. See the [README](../README.md#usage). + +## Usage + +1. Add Kubespray to your requirements.yml file + + ```yaml + collections: + - name: https://github.com/kubernetes-sigs/kubespray + type: git + version: v2.23.0 + ``` + +2. Install your collection + + ```ShellSession + ansible-galaxy install -r requirements.yml + ``` + +3. Create a playbook to install your Kubernetes cluster + + ```yaml + - name: Install Kubernetes + ansible.builtin.import_playbook: kubernetes_sigs.kubespray.cluster + ``` + +4. Update INVENTORY and PLAYBOOK so that they point to your inventory file and the playbook you created above, and then install Kubespray + + ```ShellSession + ansible-playbook -i INVENTORY --become --become-user=root PLAYBOOK + ``` diff --git a/kubespray/docs/arch.md b/kubespray/docs/arch.md new file mode 100644 index 0000000..0c91f5c --- /dev/null +++ b/kubespray/docs/arch.md @@ -0,0 +1,17 @@ +# Architecture compatibility + +The following table shows the impact of the CPU architecture on compatible features: + +- amd64: Cluster using only x86/amd64 CPUs +- arm64: Cluster using only arm64 CPUs +- amd64 + arm64: Cluster with a mix of x86/amd64 and arm64 CPUs + +| kube_network_plugin | amd64 | arm64 | amd64 + arm64 | +|---------------------|-------|-------|---------------| +| Calico | Y | Y | Y | +| Weave | Y | Y | Y | +| Flannel | Y | N | N | +| Canal | Y | N | N | +| Cilium | Y | Y | N | +| Contib | Y | N | N | +| kube-router | Y | N | N | diff --git a/kubespray/docs/aws-ebs-csi.md b/kubespray/docs/aws-ebs-csi.md new file mode 100644 index 0000000..1957277 --- /dev/null +++ b/kubespray/docs/aws-ebs-csi.md @@ -0,0 +1,87 @@ +# AWS EBS CSI Driver + +AWS EBS CSI driver allows you to provision EBS volumes for pods in EC2 instances. The old in-tree AWS cloud provider is deprecated and will be removed in future versions of Kubernetes. So transitioning to the CSI driver is advised. + +To enable AWS EBS CSI driver, uncomment the `aws_ebs_csi_enabled` option in `group_vars/all/aws.yml` and set it to `true`. + +To set the number of replicas for the AWS CSI controller, you can change `aws_ebs_csi_controller_replicas` option in `group_vars/all/aws.yml`. + +Make sure to add a role, for your EC2 instances hosting Kubernetes, that allows it to do the actions necessary to request a volume and attach it: [AWS CSI Policy](https://github.com/kubernetes-sigs/aws-ebs-csi-driver/blob/master/docs/example-iam-policy.json) + +If you want to deploy the AWS EBS storage class used with the CSI Driver, you should set `persistent_volumes_enabled` in `group_vars/k8s_cluster/k8s_cluster.yml` to `true`. + +You can now run the kubespray playbook (cluster.yml) to deploy Kubernetes over AWS EC2 with EBS CSI Driver enabled. + +## Usage example + +To check if AWS EBS CSI Driver is deployed properly, check that the ebs-csi pods are running: + +```ShellSession +$ kubectl -n kube-system get pods | grep ebs +ebs-csi-controller-85d86bccc5-8gtq5 4/4 Running 4 40s +ebs-csi-node-n4b99 3/3 Running 3 40s +``` + +Check the associated storage class (if you enabled persistent_volumes): + +```ShellSession +$ kubectl get storageclass +NAME PROVISIONER AGE +ebs-sc ebs.csi.aws.com 45s +``` + +You can run a PVC and an example Pod using this file `ebs-pod.yml`: + +```yml +-- +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + name: ebs-claim +spec: + accessModes: + - ReadWriteOnce + storageClassName: ebs-sc + resources: + requests: + storage: 1Gi +--- +apiVersion: v1 +kind: Pod +metadata: + name: app +spec: + containers: + - name: app + image: centos + command: ["/bin/sh"] + args: ["-c", "while true; do echo $(date -u) >> /data/out.txt; sleep 5; done"] + volumeMounts: + - name: persistent-storage + mountPath: /data + volumes: + - name: persistent-storage + persistentVolumeClaim: + claimName: ebs-claim +``` + +Apply this conf to your cluster: ```kubectl apply -f ebs-pod.yml``` + +You should see the PVC provisioned and bound: + +```ShellSession +$ kubectl get pvc +NAME STATUS VOLUME CAPACITY ACCESS MODES STORAGECLASS AGE +ebs-claim Bound pvc-0034cb9e-1ddd-4b3f-bb9e-0b5edbf5194c 1Gi RWO ebs-sc 50s +``` + +And the volume mounted to the example Pod (wait until the Pod is Running): + +```ShellSession +$ kubectl exec -it app -- df -h | grep data +/dev/nvme1n1 1014M 34M 981M 4% /data +``` + +## More info + +For further information about the AWS EBS CSI Driver, you can refer to this page: [AWS EBS Driver](https://github.com/kubernetes-sigs/aws-ebs-csi-driver/). diff --git a/kubespray/docs/aws.md b/kubespray/docs/aws.md new file mode 100644 index 0000000..4593898 --- /dev/null +++ b/kubespray/docs/aws.md @@ -0,0 +1,93 @@ +# AWS + +To deploy kubespray on [AWS](https://aws.amazon.com/) uncomment the `cloud_provider` option in `group_vars/all.yml` and set it to `'aws'`. Refer to the [Kubespray Configuration](#kubespray-configuration) for customizing the provider. + +Prior to creating your instances, you **must** ensure that you have created IAM roles and policies for both "kubernetes-master" and "kubernetes-node". You can find the IAM policies [here](https://github.com/kubernetes-sigs/kubespray/tree/master/contrib/aws_iam/). See the [IAM Documentation](https://aws.amazon.com/documentation/iam/) if guidance is needed on how to set these up. When you bring your instances online, associate them with the respective IAM role. Nodes that are only to be used for Etcd do not need a role. + +You would also need to tag the resources in your VPC accordingly for the aws provider to utilize them. Tag the subnets, route tables and all instances that kubernetes will be run on with key `kubernetes.io/cluster/$cluster_name` (`$cluster_name` must be a unique identifier for the cluster). Tag the subnets that must be targeted by external ELBs with the key `kubernetes.io/role/elb` and internal ELBs with the key `kubernetes.io/role/internal-elb`. + +Make sure your VPC has both DNS Hostnames support and Private DNS enabled. + +The next step is to make sure the hostnames in your `inventory` file are identical to your internal hostnames in AWS. This may look something like `ip-111-222-333-444.us-west-2.compute.internal`. You can then specify how Ansible connects to these instances with `ansible_ssh_host` and `ansible_ssh_user`. + +You can now create your cluster! + +## Dynamic Inventory + +There is also a dynamic inventory script for AWS that can be used if desired. However, be aware that it makes some certain assumptions about how you'll create your inventory. It also does not handle all use cases and groups that we may use as part of more advanced deployments. Additions welcome. + +This will produce an inventory that is passed into Ansible that looks like the following: + +```json +{ + "_meta": { + "hostvars": { + "ip-172-31-3-xxx.us-east-2.compute.internal": { + "ansible_ssh_host": "172.31.3.xxx" + }, + "ip-172-31-8-xxx.us-east-2.compute.internal": { + "ansible_ssh_host": "172.31.8.xxx" + } + } + }, + "etcd": [ + "ip-172-31-3-xxx.us-east-2.compute.internal" + ], + "k8s_cluster": { + "children": [ + "kube_control_plane", + "kube_node" + ] + }, + "kube_control_plane": [ + "ip-172-31-3-xxx.us-east-2.compute.internal" + ], + "kube_node": [ + "ip-172-31-8-xxx.us-east-2.compute.internal" + ] +} +``` + +Guide: + +- Create instances in AWS as needed. +- Either during or after creation, add tags to the instances with a key of `kubespray-role` and a value of `kube_control_plane`, `etcd`, or `kube_node`. You can also share roles like `kube_control_plane, etcd` +- Copy the `kubespray-aws-inventory.py` script from `kubespray/contrib/aws_inventory` to the `kubespray/inventory` directory. +- Set the following AWS credentials and info as environment variables in your terminal: + +```ShellSession +export AWS_ACCESS_KEY_ID="xxxxx" +export AWS_SECRET_ACCESS_KEY="yyyyy" +export AWS_REGION="us-east-2" +``` + +- We will now create our cluster. There will be either one or two small changes. The first is that we will specify `-i inventory/kubespray-aws-inventory.py` as our inventory script. The other is conditional. If your AWS instances are public facing, you can set the `VPC_VISIBILITY` variable to `public` and that will result in public IP and DNS names being passed into the inventory. This causes your cluster.yml command to look like `VPC_VISIBILITY="public" ansible-playbook ... cluster.yml` + +**Optional** Using labels and taints + +To add labels to your kubernetes node, add the following tag to your instance: + +- Key: `kubespray-node-labels` +- Value: `node-role.kubernetes.io/ingress=` + +To add taints to your kubernetes node, add the following tag to your instance: + +- Key: `kubespray-node-taints` +- Value: `node-role.kubernetes.io/ingress=:NoSchedule` + +## Kubespray configuration + +Declare the cloud config variables for the `aws` provider as follows. Setting these variables are optional and depend on your use case. + +| Variable | Type | Comment | +|------------------------------------|--------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| aws_zone | string | Force set the AWS zone. Recommended to leave blank. | +| aws_vpc | string | The AWS VPC flag enables the possibility to run the master components on a different aws account, on a different cloud provider or on-premise. If the flag is set also the KubernetesClusterTag must be provided | +| aws_subnet_id | string | SubnetID enables using a specific subnet to use for ELB's | +| aws_route_table_id | string | RouteTableID enables using a specific RouteTable | +| aws_role_arn | string | RoleARN is the IAM role to assume when interaction with AWS APIs | +| aws_kubernetes_cluster_tag | string | KubernetesClusterTag is the legacy cluster id we'll use to identify our cluster resources | +| aws_kubernetes_cluster_id | string | KubernetesClusterID is the cluster id we'll use to identify our cluster resources | +| aws_disable_security_group_ingress | bool | The aws provider creates an inbound rule per load balancer on the node security group. However, this can run into the AWS security group rule limit of 50 if many LoadBalancers are created. This flag disables the automatic ingress creation. It requires that the user has setup a rule that allows inbound traffic on kubelet ports from the local VPC subnet (so load balancers can access it). E.g. 10.82.0.0/16 30000-32000. | +| aws_elb_security_group | string | Only in Kubelet version >= 1.7 : AWS has a hard limit of 500 security groups. For large clusters creating a security group for each ELB can cause the max number of security groups to be reached. If this is set instead of creating a new Security group for each ELB this security group will be used instead. | +| aws_disable_strict_zone_check | bool | During the instantiation of an new AWS cloud provider, the detected region is validated against a known set of regions. In a non-standard, AWS like environment (e.g. Eucalyptus), this check may be undesirable. Setting this to true will disable the check and provide a warning that the check was skipped. Please note that this is an experimental feature and work-in-progress for the moment. | diff --git a/kubespray/docs/azure-csi.md b/kubespray/docs/azure-csi.md new file mode 100644 index 0000000..6aa16c2 --- /dev/null +++ b/kubespray/docs/azure-csi.md @@ -0,0 +1,128 @@ +# Azure Disk CSI Driver + +The Azure Disk CSI driver allows you to provision volumes for pods with a Kubernetes deployment over Azure Cloud. The CSI driver replaces to volume provisioning done by the in-tree azure cloud provider which is deprecated. + +This documentation is an updated version of the in-tree Azure cloud provider documentation (azure.md). + +To deploy Azure Disk CSI driver, uncomment the `azure_csi_enabled` option in `group_vars/all/azure.yml` and set it to `true`. + +## Azure Disk CSI Storage Class + +If you want to deploy the Azure Disk storage class to provision volumes dynamically, you should set `persistent_volumes_enabled` in `group_vars/k8s_cluster/k8s_cluster.yml` to `true`. + +## Parameters + +Before creating the instances you must first set the `azure_csi_` variables in the `group_vars/all.yml` file. + +All values can be retrieved using the azure cli tool which can be downloaded here: + +After installation you have to run `az login` to get access to your account. + +### azure\_csi\_tenant\_id + azure\_csi\_subscription\_id + +Run `az account show` to retrieve your subscription id and tenant id: +`azure_csi_tenant_id` -> tenantId field +`azure_csi_subscription_id` -> id field + +### azure\_csi\_location + +The region your instances are located in, it can be something like `francecentral` or `norwayeast`. A full list of region names can be retrieved via `az account list-locations` + +### azure\_csi\_resource\_group + +The name of the resource group your instances are in, a list of your resource groups can be retrieved via `az group list` + +Or you can do `az vm list | grep resourceGroup` and get the resource group corresponding to the VMs of your cluster. + +The resource group name is not case-sensitive. + +### azure\_csi\_vnet\_name + +The name of the virtual network your instances are in, can be retrieved via `az network vnet list` + +### azure\_csi\_vnet\_resource\_group + +The name of the resource group your vnet is in, can be retrieved via `az network vnet list | grep resourceGroup` and get the resource group corresponding to the vnet of your cluster. + +### azure\_csi\_subnet\_name + +The name of the subnet your instances are in, can be retrieved via `az network vnet subnet list --resource-group RESOURCE_GROUP --vnet-name VNET_NAME` + +### azure\_csi\_security\_group\_name + +The name of the network security group your instances are in, can be retrieved via `az network nsg list` + +### azure\_csi\_aad\_client\_id + azure\_csi\_aad\_client\_secret + +These will have to be generated first: + +- Create an Azure AD Application with: + + ```ShellSession + az ad app create --display-name kubespray --identifier-uris http://kubespray --homepage http://kubespray.com --password CLIENT_SECRET + ``` + +Display name, identifier-uri, homepage and the password can be chosen + +Note the AppId in the output. + +- Create Service principal for the application with: + + ```ShellSession + az ad sp create --id AppId + ``` + +This is the AppId from the last command + +- Create the role assignment with: + + ```ShellSession + az role assignment create --role "Owner" --assignee http://kubespray --subscription SUBSCRIPTION_ID + ``` + +azure\_csi\_aad\_client\_id must be set to the AppId, azure\_csi\_aad\_client\_secret is your chosen secret. + +### azure\_csi\_use\_instance\_metadata + +Use instance metadata service where possible. Boolean value. + +## Test the Azure Disk CSI driver + +To test the dynamic provisioning using Azure CSI driver, make sure to have the storage class deployed (through persistent volumes), and apply the following manifest: + +```yml +--- +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + name: pvc-azuredisk +spec: + accessModes: + - ReadWriteOnce + resources: + requests: + storage: 1Gi + storageClassName: disk.csi.azure.com +--- +kind: Pod +apiVersion: v1 +metadata: + name: nginx-azuredisk +spec: + nodeSelector: + kubernetes.io/os: linux + containers: + - image: nginx + name: nginx-azuredisk + command: + - "/bin/sh" + - "-c" + - while true; do echo $(date) >> /mnt/azuredisk/outfile; sleep 1; done + volumeMounts: + - name: azuredisk + mountPath: "/mnt/azuredisk" + volumes: + - name: azuredisk + persistentVolumeClaim: + claimName: pvc-azuredisk +``` diff --git a/kubespray/docs/azure.md b/kubespray/docs/azure.md new file mode 100644 index 0000000..a164ea7 --- /dev/null +++ b/kubespray/docs/azure.md @@ -0,0 +1,123 @@ +# Azure + +To deploy Kubernetes on [Azure](https://azure.microsoft.com) uncomment the `cloud_provider` option in `group_vars/all/all.yml` and set it to `'azure'`. + +All your instances are required to run in a resource group and a routing table has to be attached to the subnet your instances are in. + +Not all features are supported yet though, for a list of the current status have a look [here](https://github.com/Azure/AKS) + +## Parameters + +Before creating the instances you must first set the `azure_` variables in the `group_vars/all/all.yml` file. + +All values can be retrieved using the Azure CLI tool which can be downloaded here: +After installation you have to run `az login` to get access to your account. + +### azure_cloud + +Azure Stack has different API endpoints, depending on the Azure Stack deployment. These need to be provided to the Azure SDK. +Possible values are: `AzureChinaCloud`, `AzureGermanCloud`, `AzurePublicCloud` and `AzureUSGovernmentCloud`. +The full list of existing settings for the AzureChinaCloud, AzureGermanCloud, AzurePublicCloud and AzureUSGovernmentCloud +is available in the source code [here](https://github.com/kubernetes-sigs/cloud-provider-azure/blob/master/docs/cloud-provider-config.md) + +### azure\_tenant\_id + azure\_subscription\_id + +run `az account show` to retrieve your subscription id and tenant id: +`azure_tenant_id` -> Tenant ID field +`azure_subscription_id` -> ID field + +### azure\_location + +The region your instances are located, can be something like `westeurope` or `westcentralus`. A full list of region names can be retrieved via `az account list-locations` + +### azure\_resource\_group + +The name of the resource group your instances are in, can be retrieved via `az group list` + +### azure\_vmtype + +The type of the vm. Supported values are `standard` or `vmss`. If vm is type of `Virtual Machines` then value is `standard`. If vm is part of `Virtual Machine Scale Sets` then value is `vmss` + +### azure\_vnet\_name + +The name of the virtual network your instances are in, can be retrieved via `az network vnet list` + +### azure\_vnet\_resource\_group + +The name of the resource group that contains the vnet. + +### azure\_subnet\_name + +The name of the subnet your instances are in, can be retrieved via `az network vnet subnet list --resource-group RESOURCE_GROUP --vnet-name VNET_NAME` + +### azure\_security\_group\_name + +The name of the network security group your instances are in, can be retrieved via `az network nsg list` + +### azure\_security\_group\_resource\_group + +The name of the resource group that contains the network security group. Defaults to `azure_vnet_resource_group` + +### azure\_route\_table\_name + +The name of the route table used with your instances. + +### azure\_route\_table\_resource\_group + +The name of the resource group that contains the route table. Defaults to `azure_vnet_resource_group` + +### azure\_aad\_client\_id + azure\_aad\_client\_secret + +These will have to be generated first: + +- Create an Azure AD Application with: + + ```ShellSession + az ad app create --display-name kubernetes --identifier-uris http://kubernetes --homepage http://example.com --password CLIENT_SECRET + ``` + +display name, identifier-uri, homepage and the password can be chosen +Note the AppId in the output. + +- Create Service principal for the application with: + + ```ShellSession + az ad sp create --id AppId + ``` + +This is the AppId from the last command + +- Create the role assignment with: + + ```ShellSession + az role assignment create --role "Owner" --assignee http://kubernetes --subscription SUBSCRIPTION_ID + ``` + +azure\_aad\_client\_id must be set to the AppId, azure\_aad\_client\_secret is your chosen secret. + +### azure\_loadbalancer\_sku + +Sku of Load Balancer and Public IP. Candidate values are: basic and standard. + +### azure\_exclude\_master\_from\_standard\_lb + +azure\_exclude\_master\_from\_standard\_lb excludes master nodes from `standard` load balancer. + +### azure\_disable\_outbound\_snat + +azure\_disable\_outbound\_snat disables the outbound SNAT for public load balancer rules. It should only be set when azure\_exclude\_master\_from\_standard\_lb is `standard`. + +### azure\_primary\_availability\_set\_name + +(Optional) The name of the availability set that should be used as the load balancer backend .If this is set, the Azure +cloudprovider will only add nodes from that availability set to the load balancer backend pool. If this is not set, and +multiple agent pools (availability sets) are used, then the cloudprovider will try to add all nodes to a single backend +pool which is forbidden. In other words, if you use multiple agent pools (availability sets), you MUST set this field. + +### azure\_use\_instance\_metadata + +Use instance metadata service where possible + +## Provisioning Azure with Resource Group Templates + +You'll find Resource Group Templates and scripts to provision the required infrastructure to Azure in [*contrib/azurerm*](../contrib/azurerm/README.md) diff --git a/kubespray/docs/bootstrap-os.md b/kubespray/docs/bootstrap-os.md new file mode 100644 index 0000000..c2a75c0 --- /dev/null +++ b/kubespray/docs/bootstrap-os.md @@ -0,0 +1,61 @@ +# bootstrap-os + +Bootstrap an Ansible host to be able to run Ansible modules. + +This role will: + +* configure the package manager (if applicable) to be able to fetch packages +* install Python +* install the necessary packages to use Ansible's package manager modules +* set the hostname of the host to `{{ inventory_hostname }}` when requested + +## Requirements + +A host running an operating system that is supported by Kubespray. +See [Supported Linux Distributions](https://github.com/kubernetes-sigs/kubespray#supported-linux-distributions) for a current list. + +SSH access to the host. + +## Role Variables + +Variables are listed with their default values, if applicable. + +### General variables + +* `http_proxy`/`https_proxy` + The role will configure the package manager (if applicable) to download packages via a proxy. + +* `override_system_hostname: true` + The role will set the hostname of the machine to the name it has according to Ansible's inventory (the variable `{{ inventory_hostname }}`). + +### Per distribution variables + +#### Flatcar Container Linux + +* `coreos_locksmithd_disable: false` + Whether `locksmithd` (responsible for rolling restarts) should be disabled or be left alone. + +#### CentOS/RHEL/AlmaLinux/Rocky Linux + +* `centos_fastestmirror_enabled: false` + Whether the [fastestmirror](https://wiki.centos.org/PackageManagement/Yum/FastestMirror) yum plugin should be enabled. + +## Dependencies + +The `kubespray-defaults` role is expected to be run before this role. + +## Example Playbook + +Remember to disable fact gathering since Python might not be present on hosts. + +```yaml +- hosts: all + gather_facts: false # not all hosts might be able to run modules yet + roles: + - kubespray-defaults + - bootstrap-os +``` + +## License + +Apache 2.0 diff --git a/kubespray/docs/calico.md b/kubespray/docs/calico.md new file mode 100644 index 0000000..ce9432c --- /dev/null +++ b/kubespray/docs/calico.md @@ -0,0 +1,426 @@ +# Calico + +Check if the calico-node container is running + +```ShellSession +docker ps | grep calico +``` + +The **calicoctl.sh** is wrap script with configured access credentials for command calicoctl allows to check the status of the network workloads. + +* Check the status of Calico nodes + +```ShellSession +calicoctl.sh node status +``` + +* Show the configured network subnet for containers + +```ShellSession +calicoctl.sh get ippool -o wide +``` + +* Show the workloads (ip addresses of containers and their location) + +```ShellSession +calicoctl.sh get workloadEndpoint -o wide +``` + +and + +```ShellSession +calicoctl.sh get hostEndpoint -o wide +``` + +## Configuration + +### Optional : Define datastore type + +The default datastore, Kubernetes API datastore is recommended for on-premises deployments, and supports only Kubernetes workloads; etcd is the best datastore for hybrid deployments. + +Allowed values are `kdd` (default) and `etcd`. + +Note: using kdd and more than 50 nodes, consider using the `typha` daemon to provide scaling. + +To re-define you need to edit the inventory and add a group variable `calico_datastore` + +```yml +calico_datastore: kdd +``` + +### Optional : Define network backend + +In some cases you may want to define Calico network backend. Allowed values are `bird`, `vxlan` or `none`. `vxlan` is the default value. + +To re-define you need to edit the inventory and add a group variable `calico_network_backend` + +```yml +calico_network_backend: none +``` + +### Optional : Define the default pool CIDRs + +By default, `kube_pods_subnet` is used as the IP range CIDR for the default IP Pool, and `kube_pods_subnet_ipv6` for IPv6. +In some cases you may want to add several pools and not have them considered by Kubernetes as external (which means that they must be within or equal to the range defined in `kube_pods_subnet` and `kube_pods_subnet_ipv6` ), it starts with the default IP Pools of which IP range CIDRs can by defined in group_vars (k8s_cluster/k8s-net-calico.yml): + +```ShellSession +calico_pool_cidr: 10.233.64.0/20 +calico_pool_cidr_ipv6: fd85:ee78:d8a6:8607::1:0000/112 +``` + +### Optional : BGP Peering with border routers + +In some cases you may want to route the pods subnet and so NAT is not needed on the nodes. +For instance if you have a cluster spread on different locations and you want your pods to talk each other no matter where they are located. +The following variables need to be set as follow: + +```yml +peer_with_router: true # enable the peering with the datacenter's border router (default value: false). +nat_outgoing: false # (optional) NAT outgoing (default value: true). +``` + +And you'll need to edit the inventory and add a hostvar `local_as` by node. + +```ShellSession +node1 ansible_ssh_host=95.54.0.12 local_as=xxxxxx +``` + +### Optional : Defining BGP peers + +Peers can be defined using the `peers` variable (see docs/calico_peer_example examples). +In order to define global peers, the `peers` variable can be defined in group_vars with the "scope" attribute of each global peer set to "global". +In order to define peers on a per node basis, the `peers` variable must be defined in hostvars. +NB: Ansible's `hash_behaviour` is by default set to "replace", thus defining both global and per node peers would end up with having only per node peers. If having both global and per node peers defined was meant to happen, global peers would have to be defined in hostvars for each host (as well as per node peers) + +Since calico 3.4, Calico supports advertising Kubernetes service cluster IPs over BGP, just as it advertises pod IPs. +This can be enabled by setting the following variable as follow in group_vars (k8s_cluster/k8s-net-calico.yml) + +```yml +calico_advertise_cluster_ips: true +``` + +Since calico 3.10, Calico supports advertising Kubernetes service ExternalIPs over BGP in addition to cluster IPs advertising. +This can be enabled by setting the following variable in group_vars (k8s_cluster/k8s-net-calico.yml) + +```yml +calico_advertise_service_external_ips: +- x.x.x.x/24 +- y.y.y.y/32 +``` + +### Optional : Define global AS number + +Optional parameter `global_as_num` defines Calico global AS number (`/calico/bgp/v1/global/as_num` etcd key). +It defaults to "64512". + +### Optional : BGP Peering with route reflectors + +At large scale you may want to disable full node-to-node mesh in order to +optimize your BGP topology and improve `calico-node` containers' start times. + +To do so you can deploy BGP route reflectors and peer `calico-node` with them as +recommended here: + +* +* + +You need to edit your inventory and add: + +* `calico_rr` group with nodes in it. `calico_rr` can be combined with + `kube_node` and/or `kube_control_plane`. `calico_rr` group also must be a child + group of `k8s_cluster` group. +* `cluster_id` by route reflector node/group (see details [here](https://hub.docker.com/r/calico/routereflector/)) + +Here's an example of Kubespray inventory with standalone route reflectors: + +```ini +[all] +rr0 ansible_ssh_host=10.210.1.10 ip=10.210.1.10 +rr1 ansible_ssh_host=10.210.1.11 ip=10.210.1.11 +node2 ansible_ssh_host=10.210.1.12 ip=10.210.1.12 +node3 ansible_ssh_host=10.210.1.13 ip=10.210.1.13 +node4 ansible_ssh_host=10.210.1.14 ip=10.210.1.14 +node5 ansible_ssh_host=10.210.1.15 ip=10.210.1.15 + +[kube_control_plane] +node2 +node3 + +[etcd] +node2 +node3 +node4 + +[kube_node] +node2 +node3 +node4 +node5 + +[k8s_cluster:children] +kube_node +kube_control_plane +calico_rr + +[calico_rr] +rr0 +rr1 + +[rack0] +rr0 +rr1 +node2 +node3 +node4 +node5 + +[rack0:vars] +cluster_id="1.0.0.1" +calico_rr_id=rr1 +calico_group_id=rr1 +``` + +The inventory above will deploy the following topology assuming that calico's +`global_as_num` is set to `65400`: + +![Image](figures/kubespray-calico-rr.png?raw=true) + +### Optional : Define default endpoint to host action + +By default Calico blocks traffic from endpoints to the host itself by using an iptables DROP action. When using it in kubernetes the action has to be changed to RETURN (default in kubespray) or ACCEPT (see ) Otherwise all network packets from pods (with hostNetwork=False) to services endpoints (with hostNetwork=True) within the same node are dropped. + +To re-define default action please set the following variable in your inventory: + +```yml +calico_endpoint_to_host_action: "ACCEPT" +``` + +### Optional : Define address on which Felix will respond to health requests + +Since Calico 3.2.0, HealthCheck default behavior changed from listening on all interfaces to just listening on localhost. + +To re-define health host please set the following variable in your inventory: + +```yml +calico_healthhost: "0.0.0.0" +``` + +### Optional : Configure VXLAN hardware Offload + +The VXLAN Offload is disable by default. It can be configured like this to enabled it: + +```yml +calico_feature_detect_override: "ChecksumOffloadBroken=false" # The vxlan offload will enabled (It may cause problem on buggy NIC driver) +``` + +### Optional : Configure Calico Node probe timeouts + +Under certain conditions a deployer may need to tune the Calico liveness and readiness probes timeout settings. These can be configured like this: + +```yml +calico_node_livenessprobe_timeout: 10 +calico_node_readinessprobe_timeout: 10 +``` + +## Config encapsulation for cross server traffic + +Calico supports two types of encapsulation: [VXLAN and IP in IP](https://docs.projectcalico.org/v3.11/networking/vxlan-ipip). VXLAN is the more mature implementation and enabled by default, please check your environment if you need *IP in IP* encapsulation. + +*IP in IP* and *VXLAN* is mutually exclusive modes. + +Kubespray defaults have changed after version 2.18 from auto-enabling `ipip` mode to auto-enabling `vxlan`. This was done to facilitate wider deployment scenarios including those where vxlan acceleration is provided by the underlying network devices. + +If you are running your cluster with the default calico settings and are upgrading to a release post 2.18.x (i.e. 2.19 and later or `master` branch) then you have two options: + +* perform a manual migration to vxlan before upgrading kubespray (see migrating from IP in IP to VXLAN below) +* pin the pre-2.19 settings in your ansible inventory (see IP in IP mode settings below) + +**Note:**: Vxlan in ipv6 only supported when kernel >= 3.12. So if your kernel version < 3.12, Please don't set `calico_vxlan_mode_ipv6: vxlanAlways`. More details see [#Issue 6877](https://github.com/projectcalico/calico/issues/6877). + +### IP in IP mode + +To configure Ip in Ip mode you need to use the bird network backend. + +```yml +calico_ipip_mode: 'Always' # Possible values is `Always`, `CrossSubnet`, `Never` +calico_vxlan_mode: 'Never' +calico_network_backend: 'bird' +``` + +### BGP mode + +To enable BGP no-encapsulation mode: + +```yml +calico_ipip_mode: 'Never' +calico_vxlan_mode: 'Never' +calico_network_backend: 'bird' +``` + +### Migrating from IP in IP to VXLAN + +If you would like to migrate from the old IP in IP with `bird` network backends default to the new VXLAN based encapsulation you need to perform this change before running an upgrade of your cluster; the `cluster.yml` and `upgrade-cluster.yml` playbooks will refuse to continue if they detect incompatible settings. + +Execute the following steps on one of the control plane nodes, ensure the cluster in healthy before proceeding. + +```shell +calicoctl.sh patch felixconfig default -p '{"spec":{"vxlanEnabled":true}}' +calicoctl.sh patch ippool default-pool -p '{"spec":{"ipipMode":"Never", "vxlanMode":"Always"}}' +``` + +**Note:** if you created multiple ippools you will need to patch all of them individually to change their encapsulation. The kubespray playbooks only handle the default ippool created by kubespray. + +Wait for the `vxlan.calico` interfaces to be created on all cluster nodes and traffic to be routed through it then you can disable `ipip`. + +```shell +calicoctl.sh patch felixconfig default -p '{"spec":{"ipipEnabled":false}}' +``` + +## Configuring interface MTU + +This is an advanced topic and should usually not be modified unless you know exactly what you are doing. Calico is smart enough to deal with the defaults and calculate the proper MTU. If you do need to set up a custom MTU you can change `calico_veth_mtu` as follows: + +* If Wireguard is enabled, subtract 60 from your network MTU (i.e. 1500-60=1440) +* If using VXLAN or BPF mode is enabled, subtract 50 from your network MTU (i.e. 1500-50=1450) +* If using IPIP, subtract 20 from your network MTU (i.e. 1500-20=1480) +* if not using any encapsulation, set to your network MTU (i.e. 1500 or 9000) + +```yaml +calico_veth_mtu: 1440 +``` + +## Cloud providers configuration + +Please refer to the official documentation, for example [GCE configuration](http://docs.projectcalico.org/v1.5/getting-started/docker/installation/gce) requires a security rule for calico ip-ip tunnels. Note, calico is always configured with ``calico_ipip_mode: Always`` if the cloud provider was defined. + +### Optional : Ignore kernel's RPF check setting + +By default the felix agent(calico-node) will abort if the Kernel RPF setting is not 'strict'. If you want Calico to ignore the Kernel setting: + +```yml +calico_node_ignorelooserpf: true +``` + +Note that in OpenStack you must allow `ipip` traffic in your security groups, +otherwise you will experience timeouts. +To do this you must add a rule which allows it, for example: + +### Optional : Felix configuration via extraenvs of calico node + +Possible environment variable parameters for [configuring Felix](https://docs.projectcalico.org/reference/felix/configuration) + +```yml +calico_node_extra_envs: + FELIX_DEVICEROUTESOURCEADDRESS: 172.17.0.1 +``` + +```ShellSession +neutron security-group-rule-create --protocol 4 --direction egress k8s-a0tp4t +neutron security-group-rule-create --protocol 4 --direction igress k8s-a0tp4t +``` + +### Optional : Use Calico CNI host-local IPAM plugin + +Calico currently supports two types of CNI IPAM plugins, `host-local` and `calico-ipam` (default). + +To allow Calico to determine the subnet to use from the Kubernetes API based on the `Node.podCIDR` field, enable the following setting. + +```yml +calico_ipam_host_local: true +``` + +Refer to Project Calico section [Using host-local IPAM](https://docs.projectcalico.org/reference/cni-plugin/configuration#using-host-local-ipam) for further information. + +### Optional : Disable CNI logging to disk + +Calico CNI plugin logs to /var/log/calico/cni/cni.log and to stderr. +stderr of CNI plugins can be found in the logs of container runtime. + +You can disable Calico CNI logging to disk by setting `calico_cni_log_file_path: false`. + +## eBPF Support + +Calico supports eBPF for its data plane see [an introduction to the Calico eBPF Dataplane](https://www.projectcalico.org/introducing-the-calico-ebpf-dataplane/) for further information. + +Note that it is advisable to always use the latest version of Calico when using the eBPF dataplane. + +### Enabling eBPF support + +To enable the eBPF dataplane support ensure you add the following to your inventory. Note that the `kube-proxy` is incompatible with running Calico in eBPF mode and the kube-proxy should be removed from the system. + +```yaml +calico_bpf_enabled: true +``` + +**NOTE:** there is known incompatibility in using the `kernel-kvm` kernel package on Ubuntu OSes because it is missing support for `CONFIG_NET_SCHED` which is a requirement for Calico eBPF support. When using Calico eBPF with Ubuntu ensure you run the `-generic` kernel. + +### Cleaning up after kube-proxy + +Calico node cannot clean up after kube-proxy has run in ipvs mode. If you are converting an existing cluster to eBPF you will need to ensure the `kube-proxy` DaemonSet is deleted and that ipvs rules are cleaned. + +To check that kube-proxy was running in ipvs mode: + +```ShellSession +# ipvsadm -l +``` + +To clean up any ipvs leftovers: + +```ShellSession +# ipvsadm -C +``` + +### Calico access to the kube-api + +Calico node, typha and kube-controllers need to be able to talk to the kubernetes API. Please reference the [Enabling eBPF Calico Docs](https://docs.projectcalico.org/maintenance/ebpf/enabling-bpf) for guidelines on how to do this. + +Kubespray sets up the `kubernetes-services-endpoint` configmap based on the contents of the `loadbalancer_apiserver` inventory variable documented in [HA Mode](/docs/ha-mode.md). + +If no external loadbalancer is used, Calico eBPF can also use the localhost loadbalancer option. We are able to do so only if you use the same port for the localhost apiserver loadbalancer and the kube-apiserver. In this case Calico Automatic Host Endpoints need to be enabled to allow services like `coredns` and `metrics-server` to communicate with the kubernetes host endpoint. See [this blog post](https://www.projectcalico.org/securing-kubernetes-nodes-with-calico-automatic-host-endpoints/) on enabling automatic host endpoints. + +### Tunneled versus Direct Server Return + +By default Calico uses Tunneled service mode but it can use direct server return (DSR) in order to optimize the return path for a service. + +To configure DSR: + +```yaml +calico_bpf_service_mode: "DSR" +``` + +### eBPF Logging and Troubleshooting + +In order to enable Calico eBPF mode logging: + +```yaml +calico_bpf_log_level: "Debug" +``` + +To view the logs you need to use the `tc` command to read the kernel trace buffer: + +```ShellSession +tc exec bpf debug +``` + +Please see [Calico eBPF troubleshooting guide](https://docs.projectcalico.org/maintenance/troubleshoot/troubleshoot-ebpf#ebpf-program-debug-logs). + +## Wireguard Encryption + +Calico supports using Wireguard for encryption. Please see the docs on [encrypt cluster pod traffic](https://docs.projectcalico.org/security/encrypt-cluster-pod-traffic). + +To enable wireguard support: + +```yaml +calico_wireguard_enabled: true +``` + +The following OSes will require enabling the EPEL repo in order to bring in wireguard tools: + +* CentOS 7 & 8 +* AlmaLinux 8 +* Rocky Linux 8 +* Amazon Linux 2 + +```yaml +epel_enabled: true +``` diff --git a/kubespray/docs/calico_peer_example/new-york.yml b/kubespray/docs/calico_peer_example/new-york.yml new file mode 100644 index 0000000..af497a9 --- /dev/null +++ b/kubespray/docs/calico_peer_example/new-york.yml @@ -0,0 +1,12 @@ +# --- +# peers: +# - router_id: "10.99.0.34" +# as: "65xxx" +# sourceaddress: "None" +# - router_id: "10.99.0.35" +# as: "65xxx" +# sourceaddress: "None" + +# loadbalancer_apiserver: +# address: "10.99.0.44" +# port: "8383" diff --git a/kubespray/docs/calico_peer_example/paris.yml b/kubespray/docs/calico_peer_example/paris.yml new file mode 100644 index 0000000..1768e03 --- /dev/null +++ b/kubespray/docs/calico_peer_example/paris.yml @@ -0,0 +1,12 @@ +# --- +# peers: +# - router_id: "10.99.0.2" +# as: "65xxx" +# sourceaddress: "None" +# - router_id: "10.99.0.3" +# as: "65xxx" +# sourceaddress: "None" + +# loadbalancer_apiserver: +# address: "10.99.0.21" +# port: "8383" diff --git a/kubespray/docs/centos.md b/kubespray/docs/centos.md new file mode 100644 index 0000000..4b6b733 --- /dev/null +++ b/kubespray/docs/centos.md @@ -0,0 +1,12 @@ +# CentOS and derivatives + +## CentOS 7 + +The maximum python version officially supported in CentOS is 3.6. Ansible as of version 5 (ansible core 2.12.x) increased their python requirement to python 3.8 and above. +Kubespray supports multiple ansible versions but only the default (5.x) gets wide testing coverage. If your deployment host is CentOS 7 it is recommended to use one of the earlier versions still supported. + +## CentOS 8 + +If you have containers that are using iptables in the host network namespace (`hostNetwork=true`), +you need to ensure they are using iptables-nft. +An example how k8s do the autodetection can be found [in this PR](https://github.com/kubernetes/kubernetes/pull/82966) diff --git a/kubespray/docs/cert_manager.md b/kubespray/docs/cert_manager.md new file mode 100644 index 0000000..7053175 --- /dev/null +++ b/kubespray/docs/cert_manager.md @@ -0,0 +1,196 @@ +# Installation Guide + +- [Installation Guide](#installation-guide) + - [Kubernetes TLS Root CA Certificate/Key Secret](#kubernetes-tls-root-ca-certificatekey-secret) + - [Securing Ingress Resources](#securing-ingress-resources) + - [Create New TLS Root CA Certificate and Key](#create-new-tls-root-ca-certificate-and-key) + - [Install Cloudflare PKI/TLS `cfssl` Toolkit.](#install-cloudflare-pkitls-cfssl-toolkit) + - [Create Root Certificate Authority (CA) Configuration File](#create-root-certificate-authority-ca-configuration-file) + - [Create Certficate Signing Request (CSR) Configuration File](#create-certficate-signing-request-csr-configuration-file) + - [Create TLS Root CA Certificate and Key](#create-tls-root-ca-certificate-and-key) + +Cert-Manager is a native Kubernetes certificate management controller. It can help with issuing certificates from a variety of sources, such as Let’s Encrypt, HashiCorp Vault, Venafi, a simple signing key pair, or self signed. It will ensure certificates are valid and up to date, and attempt to renew certificates at a configured time before expiry. + +## Kubernetes TLS Root CA Certificate/Key Secret + +If you're planning to secure your ingress resources using TLS client certificates, you'll need to create and deploy the Kubernetes `ca-key-pair` secret consisting of the Root CA certificate and key to your K8s cluster. + +For further information, read the official [Cert-Manager CA Configuration](https://cert-manager.io/docs/configuration/ca/) doc. + +`cert-manager` can now be enabled by editing your K8s cluster addons inventory e.g. `inventory\sample\group_vars\k8s_cluster\addons.yml` and setting `cert_manager_enabled` to true. + +```ini +# Cert manager deployment +cert_manager_enabled: true +``` + +If you don't have a TLS Root CA certificate and key available, you can create these by following the steps outlined in section [Create New TLS Root CA Certificate and Key](#create-new-tls-root-ca-certificate-and-key) using the Cloudflare PKI/TLS `cfssl` toolkit. TLS Root CA certificates and keys can also be created using `ssh-keygen` and OpenSSL, if `cfssl` is not available. + +## Securing Ingress Resources + +A common use-case for cert-manager is requesting TLS signed certificates to secure your ingress resources. This can be done by simply adding annotations to your Ingress resources and cert-manager will facilitate creating the Certificate resource for you. A small sub-component of cert-manager, ingress-shim, is responsible for this. + +To enable the Nginx Ingress controller as part of your Kubespray deployment, simply edit your K8s cluster addons inventory e.g. `inventory\sample\group_vars\k8s_cluster\addons.yml` and set `ingress_nginx_enabled` to true. + +```ini +# Nginx ingress controller deployment +ingress_nginx_enabled: true +``` + +For example, if you're using the Nginx ingress controller, you can secure the Prometheus ingress by adding the annotation `cert-manager.io/cluster-issuer: ca-issuer` and the `spec.tls` section to the `Ingress` resource definition. + +```yaml +apiVersion: networking.k8s.io/v1 +kind: Ingress +metadata: + name: prometheus-k8s + namespace: monitoring + labels: + prometheus: k8s + annotations: + kubernetes.io/ingress.class: "nginx" + cert-manager.io/cluster-issuer: ca-issuer +spec: + tls: + - hosts: + - prometheus.example.com + secretName: prometheus-dashboard-certs + rules: + - host: prometheus.example.com + http: + paths: + - path: / + pathType: ImplementationSpecific + backend: + service: + name: prometheus-k8s + port: + name: web +``` + +Once deployed to your K8s cluster, every 3 months cert-manager will automatically rotate the Prometheus `prometheus.example.com` TLS client certificate and key, and store these as the Kubernetes `prometheus-dashboard-certs` secret. + +Please consult the official upstream documentation: + +- [cert-manager Ingress Usage](https://cert-manager.io/v1.5-docs/usage/ingress/) +- [cert-manager Ingress Tutorial](https://cert-manager.io/v1.5-docs/tutorials/acme/ingress/#step-3-assign-a-dns-name) + +### ACME + +The ACME Issuer type represents a single account registered with the Automated Certificate Management Environment (ACME) Certificate Authority server. When you create a new ACME Issuer, cert-manager will generate a private key which is used to identify you with the ACME server. + +Certificates issued by public ACME servers are typically trusted by client’s computers by default. This means that, for example, visiting a website that is backed by an ACME certificate issued for that URL, will be trusted by default by most client’s web browsers. ACME certificates are typically free. + +- [ACME Configuration](https://cert-manager.io/v1.5-docs/configuration/acme/) +- [ACME HTTP Validation](https://cert-manager.io/v1.5-docs/tutorials/acme/http-validation/) + - [HTTP01 Challenges](https://cert-manager.io/v1.5-docs/configuration/acme/http01/) +- [ACME DNS Validation](https://cert-manager.io/v1.5-docs/tutorials/acme/dns-validation/) + - [DNS01 Challenges](https://cert-manager.io/v1.5-docs/configuration/acme/dns01/) +- [ACME FAQ](https://cert-manager.io/v1.5-docs/faq/acme/) + +#### ACME With An Internal Certificate Authority + +The ACME Issuer with an internal certificate authority requires cert-manager to trust the certificate authority. This trust must be done at the cert-manager deployment level. +To add a trusted certificate authority to cert-manager, add it's certificate to `group_vars/k8s-cluster/addons.yml`: + +```yaml +cert_manager_trusted_internal_ca: | + -----BEGIN CERTIFICATE----- + [REPLACE with your CA certificate] + -----END CERTIFICATE----- +``` + +Once the CA is trusted, you can define your issuer normally. + +### Create New TLS Root CA Certificate and Key + +#### Install Cloudflare PKI/TLS `cfssl` Toolkit + +e.g. For Ubuntu/Debian distributions, the toolkit is part of the `golang-cfssl` package. + +```shell +sudo apt-get install -y golang-cfssl +``` + +#### Create Root Certificate Authority (CA) Configuration File + +The default TLS certificate expiry time period is `8760h` which is 1 years from the date the certificate is created. + +```shell +$ cat > ca-config.json < ca-csr.json < The Cilium Operator is responsible for managing duties in the cluster which should logically be handled once for the entire cluster, rather than once for each node in the cluster. + +### Adding custom flags to the Cilium Operator + +You can set additional cilium-operator container arguments using `cilium_operator_custom_args`. +This is an advanced option, and you should only use it if you know what you are doing. + +Accepts an array or a string. + +```yml +cilium_operator_custom_args: ["--foo=bar", "--baz=qux"] +``` + +or + +```yml +cilium_operator_custom_args: "--foo=bar" +``` + +You do not need to add a custom flag to enable debugging. Instead, feel free to use the `CILIUM_DEBUG` variable. + +### Adding extra volumes and mounting them + +You can use `cilium_operator_extra_volumes` to add extra volumes to the Cilium Operator, and use `cilium_operator_extra_volume_mounts` to mount those volumes. +This is an advanced option, and you should only use it if you know what you are doing. + +```yml +cilium_operator_extra_volumes: + - configMap: + name: foo + name: foo-mount-path + +cilium_operator_extra_volume_mounts: + - mountPath: /tmp/foo/bar + name: foo-mount-path + readOnly: true +``` + +## Choose Cilium version + +```yml +cilium_version: v1.12.1 +``` + +## Add variable to config + +Use following variables: + +Example: + +```yml +cilium_config_extra_vars: + enable-endpoint-routes: true +``` + +## Change Identity Allocation Mode + +Cilium assigns an identity for each endpoint. This identity is used to enforce basic connectivity between endpoints. + +Cilium currently supports two different identity allocation modes: + +- "crd" stores identities in kubernetes as CRDs (custom resource definition). + - These can be queried with `kubectl get ciliumid` +- "kvstore" stores identities in an etcd kvstore. + +## Enable Transparent Encryption + +Cilium supports the transparent encryption of Cilium-managed host traffic and +traffic between Cilium-managed endpoints either using IPsec or Wireguard. + +Wireguard option is only available in Cilium 1.10.0 and newer. + +### IPsec Encryption + +For further information, make sure to check the official [Cilium documentation.](https://docs.cilium.io/en/stable/gettingstarted/encryption-ipsec/) + +To enable IPsec encryption, you just need to set three variables. + +```yml +cilium_encryption_enabled: true +cilium_encryption_type: "ipsec" +``` + +The third variable is `cilium_ipsec_key.` You need to create a secret key string for this variable. +Kubespray does not automate this process. +Cilium documentation currently recommends creating a key using the following command: + +```shell +echo "3 rfc4106(gcm(aes)) $(echo $(dd if=/dev/urandom count=20 bs=1 2> /dev/null | xxd -p -c 64)) 128" +``` + +Note that Kubespray handles secret creation. So you only need to pass the key as the `cilium_ipsec_key` variable. + +### Wireguard Encryption + +For further information, make sure to check the official [Cilium documentation.](https://docs.cilium.io/en/stable/gettingstarted/encryption-wireguard/) + +To enable Wireguard encryption, you just need to set two variables. + +```yml +cilium_encryption_enabled: true +cilium_encryption_type: "wireguard" +``` + +Kubespray currently supports Linux distributions with Wireguard Kernel mode on Linux 5.6 and newer. + +## Bandwidth Manager + +Cilium’s bandwidth manager supports the kubernetes.io/egress-bandwidth Pod annotation. + +Bandwidth enforcement currently does not work in combination with L7 Cilium Network Policies. +In case they select the Pod at egress, then the bandwidth enforcement will be disabled for those Pods. + +Bandwidth Manager requires a v5.1.x or more recent Linux kernel. + +For further information, make sure to check the official [Cilium documentation.](https://docs.cilium.io/en/v1.12/gettingstarted/bandwidth-manager/) + +To use this function, set the following parameters + +```yml +cilium_enable_bandwidth_manager: true +``` + +## Install Cilium Hubble + +k8s-net-cilium.yml: + +```yml +cilium_enable_hubble: true ## enable support hubble in cilium +cilium_hubble_install: true ## install hubble-relay, hubble-ui +cilium_hubble_tls_generate: true ## install hubble-certgen and generate certificates +``` + +To validate that Hubble UI is properly configured, set up a port forwarding for hubble-ui service: + +```shell script +kubectl port-forward -n kube-system svc/hubble-ui 12000:80 +``` + +and then open [http://localhost:12000/](http://localhost:12000/). + +## Hubble metrics + +```yml +cilium_enable_hubble_metrics: true +cilium_hubble_metrics: + - dns + - drop + - tcp + - flow + - icmp + - http +``` + +[More](https://docs.cilium.io/en/v1.9/operations/metrics/#hubble-exported-metrics) + +## Upgrade considerations + +### Rolling-restart timeouts + +Cilium relies on the kernel's BPF support, which is extremely fast at runtime but incurs a compilation penalty on initialization and update. + +As a result, the Cilium DaemonSet pods can take a significant time to start, which scales with the number of nodes and endpoints in your cluster. + +As part of cluster.yml, this DaemonSet is restarted, and Kubespray's [default timeouts for this operation](../roles/network_plugin/cilium/defaults/main.yml) +are not appropriate for large clusters. + +This means that you will likely want to update these timeouts to a value more in-line with your cluster's number of nodes and their respective CPU performance. +This is configured by the following values: + +```yaml +# Configure how long to wait for the Cilium DaemonSet to be ready again +cilium_rolling_restart_wait_retries_count: 30 +cilium_rolling_restart_wait_retries_delay_seconds: 10 +``` + +The total time allowed (count * delay) should be at least `($number_of_nodes_in_cluster * $cilium_pod_start_time)` for successful rolling updates. There are no +drawbacks to making it higher and giving yourself a time buffer to accommodate transient slowdowns. + +Note: To find the `$cilium_pod_start_time` for your cluster, you can simply restart a Cilium pod on a node of your choice and look at how long it takes for it +to become ready. + +Note 2: The default CPU requests/limits for Cilium pods is set to a very conservative 100m:500m which will likely yield very slow startup for Cilium pods. You +probably want to significantly increase the CPU limit specifically if short bursts of CPU from Cilium are acceptable to you. diff --git a/kubespray/docs/cinder-csi.md b/kubespray/docs/cinder-csi.md new file mode 100644 index 0000000..a695452 --- /dev/null +++ b/kubespray/docs/cinder-csi.md @@ -0,0 +1,102 @@ +# Cinder CSI Driver + +Cinder CSI driver allows you to provision volumes over an OpenStack deployment. The Kubernetes historic in-tree cloud provider is deprecated and will be removed in future versions. + +To enable Cinder CSI driver, uncomment the `cinder_csi_enabled` option in `group_vars/all/openstack.yml` and set it to `true`. + +To set the number of replicas for the Cinder CSI controller, you can change `cinder_csi_controller_replicas` option in `group_vars/all/openstack.yml`. + +You need to source the OpenStack credentials you use to deploy your machines that will host Kubernetes: `source path/to/your/openstack-rc` or `. path/to/your/openstack-rc`. + +Make sure the hostnames in your `inventory` file are identical to your instance names in OpenStack. Otherwise [cinder](https://docs.openstack.org/cinder/latest/) won't work as expected. + +If you want to deploy the cinder provisioner used with Cinder CSI Driver, you should set `persistent_volumes_enabled` in `group_vars/k8s_cluster/k8s_cluster.yml` to `true`. + +You can now run the kubespray playbook (cluster.yml) to deploy Kubernetes over OpenStack with Cinder CSI Driver enabled. + +## Usage example + +To check if Cinder CSI Driver works properly, see first that the cinder-csi pods are running: + +```ShellSession +$ kubectl -n kube-system get pods | grep cinder +csi-cinder-controllerplugin-7f8bf99785-cpb5v 5/5 Running 0 100m +csi-cinder-nodeplugin-rm5x2 2/2 Running 0 100m +``` + +Check the associated storage class (if you enabled persistent_volumes): + +```ShellSession +$ kubectl get storageclass +NAME PROVISIONER AGE +cinder-csi cinder.csi.openstack.org 100m +``` + +You can run a PVC and an Nginx Pod using this file `nginx.yaml`: + +```yml +--- +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + name: csi-pvc-cinderplugin +spec: + accessModes: + - ReadWriteOnce + resources: + requests: + storage: 1Gi + storageClassName: cinder-csi + +--- +apiVersion: v1 +kind: Pod +metadata: + name: nginx +spec: + containers: + - image: nginx + imagePullPolicy: IfNotPresent + name: nginx + ports: + - containerPort: 80 + protocol: TCP + volumeMounts: + - mountPath: /var/lib/www/html + name: csi-data-cinderplugin + volumes: + - name: csi-data-cinderplugin + persistentVolumeClaim: + claimName: csi-pvc-cinderplugin + readOnly: false +``` + +Apply this conf to your cluster: ```kubectl apply -f nginx.yml``` + +You should see the PVC provisioned and bound: + +```ShellSession +$ kubectl get pvc +NAME STATUS VOLUME CAPACITY ACCESS MODES STORAGECLASS AGE +csi-pvc-cinderplugin Bound pvc-f21ad0a1-5b7b-405e-a462-48da5cb76beb 1Gi RWO cinder-csi 8s +``` + +And the volume mounted to the Nginx Pod (wait until the Pod is Running): + +```ShellSession +kubectl exec -it nginx -- df -h | grep /var/lib/www/html +/dev/vdb 976M 2.6M 958M 1% /var/lib/www/html +``` + +## Compatibility with in-tree cloud provider + +It is not necessary to enable OpenStack as a cloud provider for Cinder CSI Driver to work. +Though, you can run both the in-tree openstack cloud provider and the Cinder CSI Driver at the same time. The storage class provisioners associated to each one of them are differently named. + +## Cinder v2 support + +For the moment, only Cinder v3 is supported by the CSI Driver. + +## More info + +For further information about the Cinder CSI Driver, you can refer to this page: [Cloud Provider OpenStack](https://github.com/kubernetes/cloud-provider-openstack/blob/master/docs/cinder-csi-plugin/using-cinder-csi-plugin.md). diff --git a/kubespray/docs/cloud.md b/kubespray/docs/cloud.md new file mode 100644 index 0000000..d7fcfef --- /dev/null +++ b/kubespray/docs/cloud.md @@ -0,0 +1,13 @@ +# Cloud providers + +## Provisioning + +You can deploy instances in your cloud environment in several ways. Examples include Terraform, Ansible (ec2 and gce modules), and manual creation. + +## Deploy kubernetes + +With ansible-playbook command + +```ShellSession +ansible-playbook -u smana -e ansible_ssh_user=admin -e cloud_provider=[aws|gce] -b --become-user=root -i inventory/single.cfg cluster.yml +``` diff --git a/kubespray/docs/cni.md b/kubespray/docs/cni.md new file mode 100644 index 0000000..72e331d --- /dev/null +++ b/kubespray/docs/cni.md @@ -0,0 +1,8 @@ +CNI +============== + +This network plugin only unpacks CNI plugins version `cni_version` into `/opt/cni/bin` and instructs implementation of container runtime cri to use cni. + +It's intended usage is for custom CNI configuration, e.g. manual routing tables + bridge + loopback CNI plugin outside kubespray scope. Furthermore, it's used for non-kubespray supported CNI plugins which you can install afterward. + +You are required to fill `/etc/cni/net.d` with valid CNI configuration after using kubespray. diff --git a/kubespray/docs/comparisons.md b/kubespray/docs/comparisons.md new file mode 100644 index 0000000..7ad1921 --- /dev/null +++ b/kubespray/docs/comparisons.md @@ -0,0 +1,26 @@ +# Comparison + +## Kubespray vs Kops + +Kubespray runs on bare metal and most clouds, using Ansible as its substrate for +provisioning and orchestration. [Kops](https://github.com/kubernetes/kops) performs the provisioning and orchestration +itself, and as such is less flexible in deployment platforms. For people with +familiarity with Ansible, existing Ansible deployments or the desire to run a +Kubernetes cluster across multiple platforms, Kubespray is a good choice. Kops, +however, is more tightly integrated with the unique features of the clouds it +supports so it could be a better choice if you know that you will only be using +one platform for the foreseeable future. + +## Kubespray vs Kubeadm + +[Kubeadm](https://github.com/kubernetes/kubeadm) provides domain Knowledge of Kubernetes clusters' life cycle +management, including self-hosted layouts, dynamic discovery services and so +on. Had it belonged to the new [operators world](https://coreos.com/blog/introducing-operators.html), +it may have been named a "Kubernetes cluster operator". Kubespray however, +does generic configuration management tasks from the "OS operators" ansible +world, plus some initial K8s clustering (with networking plugins included) and +control plane bootstrapping. + +Kubespray has started using `kubeadm` internally for cluster creation since v2.3 +in order to consume life cycle management domain knowledge from it +and offload generic OS configuration things from it, which hopefully benefits both sides. diff --git a/kubespray/docs/containerd.md b/kubespray/docs/containerd.md new file mode 100644 index 0000000..112c0fc --- /dev/null +++ b/kubespray/docs/containerd.md @@ -0,0 +1,142 @@ +# containerd + +[containerd] An industry-standard container runtime with an emphasis on simplicity, robustness and portability +Kubespray supports basic functionality for using containerd as the default container runtime in a cluster. + +_To use the containerd container runtime set the following variables:_ + +## k8s_cluster.yml + +When kube_node contains etcd, you define your etcd cluster to be as well schedulable for Kubernetes workloads. Thus containerd and dockerd can not run at same time, must be set to bellow for running etcd cluster with only containerd. + +```yaml +container_manager: containerd +``` + +## etcd.yml + +```yaml +etcd_deployment_type: host +``` + +## Containerd config + +Example: define registry mirror for docker hub + +```yaml +containerd_registries_mirrors: + - prefix: docker.io + mirrors: + - host: https://mirror.gcr.io + capabilities: ["pull", "resolve"] + skip_verify: false + - host: https://registry-1.docker.io + capabilities: ["pull", "resolve"] + skip_verify: false +``` + +`containerd_registries_mirrors` is ignored for pulling images when `image_command_tool=nerdctl` +(the default for `container_manager=containerd`). Use `crictl` instead, it supports +`containerd_registries_mirrors` but lacks proper multi-arch support (see +[#8375](https://github.com/kubernetes-sigs/kubespray/issues/8375)): + +```yaml +image_command_tool: crictl +``` + +### Containerd Runtimes + +Containerd supports multiple runtime configurations that can be used with +[RuntimeClass] Kubernetes feature. See [runtime classes in containerd] for the +details of containerd configuration. + +In kubespray, the default runtime name is "runc", and it can be configured with the `containerd_runc_runtime` dictionary: + +```yaml +containerd_runc_runtime: + name: runc + type: "io.containerd.runc.v2" + engine: "" + root: "" + options: + systemdCgroup: "false" + binaryName: /usr/local/bin/my-runc + base_runtime_spec: cri-base.json +``` + +Further runtimes can be configured with `containerd_additional_runtimes`, which +is a list of such dictionaries. + +Default runtime can be changed by setting `containerd_default_runtime`. + +#### Base runtime specs and limiting number of open files + +`base_runtime_spec` key in a runtime dictionary is used to explicitly +specify a runtime spec json file. `runc` runtime has it set to `cri-base.json`, +which is generated with `ctr oci spec > /etc/containerd/cri-base.json` and +updated to include a custom setting for maximum number of file descriptors per +container. + +You can change maximum number of file descriptors per container for the default +`runc` runtime by setting the `containerd_base_runtime_spec_rlimit_nofile` +variable. + +You can tune many more [settings][runtime-spec] by supplying your own file name and content with `containerd_base_runtime_specs`: + +```yaml +containerd_base_runtime_specs: + cri-spec-custom.json: | + { + "ociVersion": "1.0.2-dev", + "process": { + "user": { + "uid": 0, + ... +``` + +The files in this dict will be placed in containerd config directory, +`/etc/containerd` by default. The files can then be referenced by filename in a +runtime: + +```yaml +containerd_runc_runtime: + name: runc + base_runtime_spec: cri-spec-custom.json + ... +``` + +Config insecure-registry access to self hosted registries. + +```yaml +containerd_registries_mirrors: + - prefix: test.registry.io + mirrors: + - host: http://test.registry.io + capabilities: ["pull", "resolve"] + skip_verify: true + - prefix: 172.19.16.11:5000 + mirrors: + - host: http://172.19.16.11:5000 + capabilities: ["pull", "resolve"] + skip_verify: true + - prefix: repo:5000 + mirrors: + - host: http://repo:5000 + capabilities: ["pull", "resolve"] + skip_verify: true +``` + +[containerd]: https://containerd.io/ +[RuntimeClass]: https://kubernetes.io/docs/concepts/containers/runtime-class/ +[runtime classes in containerd]: https://github.com/containerd/containerd/blob/main/docs/cri/config.md#runtime-classes +[runtime-spec]: https://github.com/opencontainers/runtime-spec + +### Optional : NRI + +[Node Resource Interface](https://github.com/containerd/nri) (NRI) is disabled by default for the containerd. If you +are using contained version v1.7.0 or above, then you can enable it with the +following configuration: + +```yaml +nri_enabled: true +``` diff --git a/kubespray/docs/cri-o.md b/kubespray/docs/cri-o.md new file mode 100644 index 0000000..2405697 --- /dev/null +++ b/kubespray/docs/cri-o.md @@ -0,0 +1,74 @@ +# CRI-O + +[CRI-O] is a lightweight container runtime for Kubernetes. +Kubespray supports basic functionality for using CRI-O as the default container runtime in a cluster. + +* Kubernetes supports CRI-O on v1.11.1 or later. +* etcd: configure either kubeadm managed etcd or host deployment + +_To use the CRI-O container runtime set the following variables:_ + +## all/all.yml + +```yaml +download_container: false +skip_downloads: false +etcd_deployment_type: host # optionally kubeadm +``` + +## k8s_cluster/k8s_cluster.yml + +```yaml +container_manager: crio +``` + +## all/crio.yml + +Enable docker hub registry mirrors + +```yaml +crio_registries: + - prefix: docker.io + insecure: false + blocked: false + location: registry-1.docker.io + unqualified: false + mirrors: + - location: 192.168.100.100:5000 + insecure: true + - location: mirror.gcr.io + insecure: false +``` + +[CRI-O]: https://cri-o.io/ + +## Note about user namespaces + +CRI-O has support for user namespaces. This feature is optional and can be enabled by setting the following two variables. + +```yaml +crio_runtimes: + - name: runc + path: /usr/bin/runc + type: oci + root: /run/runc + allowed_annotations: + - "io.kubernetes.cri-o.userns-mode" + +crio_remap_enable: true +``` + +The `allowed_annotations` configures `crio.conf` accordingly. + +The `crio_remap_enable` configures the `/etc/subuid` and `/etc/subgid` files to add an entry for the **containers** user. +By default, 16M uids and gids are reserved for user namespaces (256 pods * 65536 uids/gids) at the end of the uid/gid space. + +## Optional : NRI + +[Node Resource Interface](https://github.com/containerd/nri) (NRI) is disabled by default for the CRI-O. If you +are using CRI-O version v1.26.0 or above, then you can enable it with the +following configuration: + +```yaml +nri_enabled: true +``` diff --git a/kubespray/docs/debian.md b/kubespray/docs/debian.md new file mode 100644 index 0000000..8c25637 --- /dev/null +++ b/kubespray/docs/debian.md @@ -0,0 +1,41 @@ +# Debian Jessie + +Debian Jessie installation Notes: + +- Add + + ```ini + GRUB_CMDLINE_LINUX="cgroup_enable=memory swapaccount=1" + ``` + + to `/etc/default/grub`. Then update with + + ```ShellSession + sudo update-grub + sudo update-grub2 + sudo reboot + ``` + +- Add the [backports](https://backports.debian.org/Instructions/) which contain Systemd 2.30 and update Systemd. + + ```ShellSession + apt-get -t jessie-backports install systemd + ``` + + (Necessary because the default Systemd version (2.15) does not support the "Delegate" directive in service files) + +- Add the Ansible repository and install Ansible to get a proper version + + ```ShellSession + sudo add-apt-repository ppa:ansible/ansible + sudo apt-get update + sudo apt-get install ansible + ``` + +- Install Jinja2 and Python-Netaddr + + ```ShellSession + sudo apt-get install python-jinja2=2.8-1~bpo8+1 python-netaddr + ``` + +Now you can continue with [Preparing your deployment](getting-started.md#starting-custom-deployment) diff --git a/kubespray/docs/dns-stack.md b/kubespray/docs/dns-stack.md new file mode 100644 index 0000000..3ba3669 --- /dev/null +++ b/kubespray/docs/dns-stack.md @@ -0,0 +1,313 @@ +# K8s DNS stack by Kubespray + +For K8s cluster nodes, Kubespray configures a [Kubernetes DNS](https://kubernetes.io/docs/tasks/administer-cluster/dns-custom-nameservers/) +[cluster add-on](https://releases.k8s.io/master/cluster/addons/README.md) +to serve as an authoritative DNS server for a given ``dns_domain`` and its +``svc, default.svc`` default subdomains (a total of ``ndots: 5`` max levels). + +Other nodes in the inventory, like external storage nodes or a separate etcd cluster +node group, considered non-cluster and left up to the user to configure DNS resolve. + +## DNS variables + +There are several global variables which can be used to modify DNS settings: + +### ndots + +ndots value to be used in ``/etc/resolv.conf`` + +It is important to note that multiple search domains combined with high ``ndots`` +values lead to poor performance of DNS stack, so please choose it wisely. + +## dns_timeout + +timeout value to be used in ``/etc/resolv.conf`` + +## dns_attempts + +attempts value to be used in ``/etc/resolv.conf`` + +### searchdomains + +Custom search domains to be added in addition to the cluster search domains (``default.svc.{{ dns_domain }}, svc.{{ dns_domain }}``). + +Most Linux systems limit the total number of search domains to 6 and the total length of all search domains +to 256 characters. Depending on the length of ``dns_domain``, you're limited to less than the total limit. + +`remove_default_searchdomains: true` will remove the default cluster search domains. + +Please note that ``resolvconf_mode: docker_dns`` will automatically add your systems search domains as +additional search domains. Please take this into the accounts for the limits. + +### nameservers + +This variable is only used by ``resolvconf_mode: host_resolvconf``. These nameservers are added to the hosts +``/etc/resolv.conf`` *after* ``upstream_dns_servers`` and thus serve as backup nameservers. If this variable +is not set, a default resolver is chosen (depending on cloud provider or 8.8.8.8 when no cloud provider is specified). + +### upstream_dns_servers + +DNS servers to be added *after* the cluster DNS. Used by all ``resolvconf_mode`` modes. These serve as backup +DNS servers in early cluster deployment when no cluster DNS is available yet. + +### dns_upstream_forward_extra_opts + +Whether or not upstream DNS servers come from `upstream_dns_servers` variable or /etc/resolv.conf, related forward block in coredns (and nodelocaldns) configuration can take options (see for details). +These are configurable in inventory in as a dictionary in the `dns_upstream_forward_extra_opts` variable. +By default, no other option than the ones hardcoded (see `roles/kubernetes-apps/ansible/templates/coredns-config.yml.j2` and `roles/kubernetes-apps/ansible/templates/nodelocaldns-config.yml.j2`). + +### coredns_kubernetes_extra_opts + +Custom options to be added to the kubernetes coredns plugin. + +### coredns_kubernetes_extra_domains + +Extra domains to be forwarded to the kubernetes coredns plugin. + +### coredns_rewrite_block + +[Rewrite](https://coredns.io/plugins/rewrite/) plugin block to perform internal message rewriting. + +### coredns_external_zones + +Array of optional external zones to coredns forward queries to. It's injected into +`coredns`' config file before default kubernetes zone. Use it as an optimization for well-known zones and/or internal-only +domains, i.e. VPN for internal networks (default is unset) + +Example: + +```yaml +coredns_external_zones: +- zones: + - example.com + - example.io:1053 + nameservers: + - 1.1.1.1 + - 2.2.2.2 + cache: 5 +- zones: + - https://mycompany.local:4453 + nameservers: + - 192.168.0.53 + cache: 0 +- zones: + - mydomain.tld + nameservers: + - 10.233.0.3 + cache: 5 + rewrite: + - name stop website.tld website.namespace.svc.cluster.local +``` + +or as INI + +```ini +coredns_external_zones='[{"cache": 30,"zones":["example.com","example.io:453"],"nameservers":["1.1.1.1","2.2.2.2"]}]' +``` + +### dns_etchosts (coredns) + +Optional hosts file content to coredns use as /etc/hosts file. This will also be used by nodelocaldns, if enabled. + +Example: + +```yaml +dns_etchosts: | + 192.168.0.100 api.example.com + 192.168.0.200 ingress.example.com +``` + +### enable_coredns_reverse_dns_lookups + +Whether reverse DNS lookups are enabled in the coredns config. Defaults to `true`. + +### CoreDNS default zone cache plugin + +If you wish to configure the caching behaviour of CoreDNS on the default zone, you can do so using the `coredns_default_zone_cache_block` string block. + +An example value (more information on the [plugin's documentation](https://coredns.io/plugins/cache/)) to: + +* raise the max cache TTL to 3600 seconds +* raise the max amount of success responses to cache to 3000 +* disable caching of denial responses altogether +* enable pre-fetching of lookups with at least 10 lookups per minute before they expire + +Would be as follows: + +```yaml +coredns_default_zone_cache_block: | + cache 3600 { + success 3000 + denial 0 + prefetch 10 1m + } +``` + +### systemd_resolved_disable_stub_listener + +Whether or not to set `DNSStubListener=no` when using systemd-resolved. Defaults to `true` on Flatcar. +You might need to set it to `true` if CoreDNS fails to start with `address already in use` errors. + +## DNS modes supported by Kubespray + +You can modify how Kubespray sets up DNS for your cluster with the variables ``dns_mode`` and ``resolvconf_mode``. + +### dns_mode + +``dns_mode`` configures how Kubespray will setup cluster DNS. There are four modes available: + +#### dns_mode: coredns (default) + +This installs CoreDNS as the default cluster DNS for all queries. + +#### dns_mode: coredns_dual + +This installs CoreDNS as the default cluster DNS for all queries, plus a secondary CoreDNS stack. + +#### dns_mode: manual + +This does not install coredns, but allows you to specify +`manual_dns_server`, which will be configured on nodes for handling Pod DNS. +Use this method if you plan to install your own DNS server in the cluster after +initial deployment. + +#### dns_mode: none + +This does not install any of DNS solution at all. This basically disables cluster DNS completely and +leaves you with a non functional cluster. + +## resolvconf_mode + +``resolvconf_mode`` configures how Kubespray will setup DNS for ``hostNetwork: true`` PODs and non-k8s containers. +There are three modes available: + +### resolvconf_mode: host_resolvconf (default) + +This activates the classic Kubespray behavior that modifies the hosts ``/etc/resolv.conf`` file and dhclient +configuration to point to the cluster dns server (either coredns or coredns_dual, depending on dns_mode). + +As cluster DNS is not available on early deployment stage, this mode is split into 2 stages. In the first +stage (``dns_early: true``), ``/etc/resolv.conf`` is configured to use the DNS servers found in ``upstream_dns_servers`` +and ``nameservers``. Later, ``/etc/resolv.conf`` is reconfigured to use the cluster DNS server first, leaving +the other nameservers as backups. + +Also note, existing records will be purged from the `/etc/resolv.conf`, +including resolvconf's base/head/cloud-init config files and those that come from dhclient. + +### resolvconf_mode: docker_dns + +This sets up the docker daemon with additional --dns/--dns-search/--dns-opt flags. + +The following nameservers are added to the docker daemon (in the same order as listed here): + +* cluster nameserver (depends on dns_mode) +* content of optional upstream_dns_servers variable +* host system nameservers (read from hosts /etc/resolv.conf) + +The following search domains are added to the docker daemon (in the same order as listed here): + +* cluster domains (``default.svc.{{ dns_domain }}``, ``svc.{{ dns_domain }}``) +* content of optional searchdomains variable +* host system search domains (read from hosts /etc/resolv.conf) + +The following dns options are added to the docker daemon + +* ndots:{{ ndots }} +* timeout:2 +* attempts:2 + +These dns options can be overridden by setting a different list: + +```yaml +docker_dns_options: +- ndots:{{ ndots }} +- timeout:2 +- attempts:2 +- rotate +``` + +For normal PODs, k8s will ignore these options and setup its own DNS settings for the PODs, taking +the --cluster_dns (either coredns or coredns_dual, depending on dns_mode) kubelet option into account. +For ``hostNetwork: true`` PODs however, k8s will let docker setup DNS settings. Docker containers which +are not started/managed by k8s will also use these docker options. + +The host system name servers are added to ensure name resolution is also working while cluster DNS is not +running yet. This is especially important in early stages of cluster deployment. In this early stage, +DNS queries to the cluster DNS will timeout after a few seconds, resulting in the system nameserver being +used as a backup nameserver. After cluster DNS is running, all queries will be answered by the cluster DNS +servers, which in turn will forward queries to the system nameserver if required. + +### resolvconf_mode: none + +Does nothing regarding ``/etc/resolv.conf``. This leaves you with a cluster that works as expected in most cases. +The only exception is that ``hostNetwork: true`` PODs and non-k8s managed containers will not be able to resolve +cluster service names. + +## Nodelocal DNS cache + +Setting ``enable_nodelocaldns`` to ``true`` will make pods reach out to the dns (core-dns) caching agent running on the same node, thereby avoiding iptables DNAT rules and connection tracking. The local caching agent will query core-dns (depending on what main DNS plugin is configured in your cluster) for cache misses of cluster hostnames(cluster.local suffix by default). + +More information on the rationale behind this implementation can be found [here](https://github.com/kubernetes/enhancements/blob/master/keps/sig-network/1024-nodelocal-cache-dns/README.md). + +**As per the 2.10 release, Nodelocal DNS cache is enabled by default.** + +### External zones + +It's possible to extent the `nodelocaldns`' configuration by adding an array of external zones. For example: + +```yaml +nodelocaldns_external_zones: +- zones: + - example.com + - example.io:1053 + nameservers: + - 1.1.1.1 + - 2.2.2.2 + cache: 5 +- zones: + - https://mycompany.local:4453 + nameservers: + - 192.168.0.53 +``` + +### dns_etchosts (nodelocaldns) + +See [dns_etchosts](#dns_etchosts-coredns) above. + +### Nodelocal DNS HA + +Under some circumstances the single POD nodelocaldns implementation may not be able to be replaced soon enough and a cluster upgrade or a nodelocaldns upgrade can cause DNS requests to time out for short intervals. If for any reason your applications cannot tolerate this behavior you can enable a redundant nodelocal DNS pod on each node: + +```yaml +enable_nodelocaldns_secondary: true +``` + +**Note:** when the nodelocaldns secondary is enabled, the primary is instructed to no longer tear down the iptables rules it sets up to direct traffic to itself. In case both daemonsets have failing pods on the same node, this can cause a DNS blackout with traffic no longer being forwarded to the coredns central service as a fallback. Please ensure you account for this also if you decide to disable the nodelocaldns cache. + +There is a time delta (in seconds) allowed for the secondary nodelocaldns to survive in case both primary and secondary daemonsets are updated at the same time. It is advised to tune this variable after you have performed some tests in your own environment. + +```yaml +nodelocaldns_secondary_skew_seconds: 5 +``` + +## Limitations + +* Kubespray has yet ways to configure Kubedns addon to forward requests SkyDns can + not answer with authority to arbitrary recursive resolvers. This task is left + for future. See [official SkyDns docs](https://github.com/skynetservices/skydns) + for details. + +* There is + [no way to specify a custom value](https://github.com/kubernetes/kubernetes/issues/33554) + for the SkyDNS ``ndots`` param. + +* the ``searchdomains`` have a limitation of a 6 names and 256 chars + length. Due to default ``svc, default.svc`` subdomains, the actual + limits are a 4 names and 239 chars respectively. If `remove_default_searchdomains: true` + added you are back to 6 names. + +* the ``nameservers`` have a limitation of a 3 servers, although there + is a way to mitigate that with the ``upstream_dns_servers``, + see below. Anyway, the ``nameservers`` can take no more than a two + custom DNS servers because of one slot is reserved for a Kubernetes + cluster needs. diff --git a/kubespray/docs/docker.md b/kubespray/docs/docker.md new file mode 100644 index 0000000..4cfcb7f --- /dev/null +++ b/kubespray/docs/docker.md @@ -0,0 +1,99 @@ +# Docker support + +The docker runtime is supported by kubespray and while the `dockershim` is deprecated to be removed in kubernetes 1.24+ there are alternative ways to use docker such as through the [cri-dockerd](https://github.com/Mirantis/cri-dockerd) project supported by Mirantis. + +Using the docker container manager: + +```yaml +container_manager: docker +``` + +*Note:* `cri-dockerd` has replaced `dockershim` across supported kubernetes version in kubespray 2.20. + +Enabling the `overlay2` graph driver: + +```yaml +docker_storage_options: -s overlay2 +``` + +Enabling `docker_container_storage_setup`, it will configure devicemapper driver on Centos7 or RedHat7. +Deployers must be define a disk path for `docker_container_storage_setup_devs`, otherwise docker-storage-setup will be executed incorrectly. + +```yaml +docker_container_storage_setup: true +docker_container_storage_setup_devs: /dev/vdb +``` + +Changing the Docker cgroup driver (native.cgroupdriver); valid options are `systemd` or `cgroupfs`, default is `systemd`: + +```yaml +docker_cgroup_driver: systemd +``` + +If you have more than 3 nameservers kubespray will only use the first 3 else it will fail. Set the `docker_dns_servers_strict` to `false` to prevent deployment failure. + +```yaml +docker_dns_servers_strict: false +``` + +Set the path used to store Docker data: + +```yaml +docker_daemon_graph: "/var/lib/docker" +``` + +Changing the docker daemon iptables support: + +```yaml +docker_iptables_enabled: "false" +``` + +Docker log options: + +```yaml +# Rotate container stderr/stdout logs at 50m and keep last 5 +docker_log_opts: "--log-opt max-size=50m --log-opt max-file=5" +``` + +Change the docker `bin_dir`, this should not be changed unless you use a custom docker package: + +```yaml +docker_bin_dir: "/usr/bin" +``` + +To keep docker packages after installation; speeds up repeated ansible provisioning runs when '1'. +kubespray deletes the docker package on each run, so caching the package makes sense: + +```yaml +docker_rpm_keepcache: 1 +``` + +Allowing insecure-registry access to self hosted registries. Can be ipaddress and domain_name. + +```yaml +## example define 172.19.16.11 or mirror.registry.io +docker_insecure_registries: + - mirror.registry.io + - 172.19.16.11 +``` + +Adding other registry, i.e. China registry mirror: + +```yaml +docker_registry_mirrors: + - https://registry.docker-cn.com + - https://mirror.aliyuncs.com +``` + +Overriding default system MountFlags value. This option takes a mount propagation flag: `shared`, `slave` or `private`, which control whether mounts in the file system namespace set up for docker will receive or propagate mounts and unmounts. Leave empty for system default: + +```yaml +docker_mount_flags: +``` + +Adding extra options to pass to the docker daemon: + +```yaml +## This string should be exactly as you wish it to appear. +docker_options: "" +``` diff --git a/kubespray/docs/downloads.md b/kubespray/docs/downloads.md new file mode 100644 index 0000000..9961eab --- /dev/null +++ b/kubespray/docs/downloads.md @@ -0,0 +1,41 @@ +# Downloading binaries and containers + +Kubespray supports several download/upload modes. The default is: + +* Each node downloads binaries and container images on its own, which is ``download_run_once: False``. +* For K8s apps, pull policy is ``k8s_image_pull_policy: IfNotPresent``. +* For system managed containers, like kubelet or etcd, pull policy is ``download_always_pull: False``, which is pull if only the wanted repo and tag/sha256 digest differs from that the host has. + +There is also a "pull once, push many" mode as well: + +* Setting ``download_run_once: True`` will make kubespray download container images and binaries only once and then push them to the cluster nodes. The default download delegate node is the first `kube_control_plane`. +* Set ``download_localhost: True`` to make localhost the download delegate. This can be useful if cluster nodes cannot access external addresses. To use this requires that the container runtime is installed and running on the Ansible master and that the current user is either in the docker group or can do passwordless sudo, to be able to use the container runtime. Note: even if `download_localhost` is false, files will still be copied to the Ansible server (local host) from the delegated download node, and then distributed from the Ansible server to all cluster nodes. + +NOTE: When `download_run_once` is true and `download_localhost` is false, all downloads will be done on the delegate node, including downloads for container images that are not required on that node. As a consequence, the storage required on that node will probably be more than if download_run_once was false, because all images will be loaded into the storage of the container runtime on that node, instead of just the images required for that node. + +On caching: + +* When `download_run_once` is `True`, all downloaded files will be cached locally in `download_cache_dir`, which defaults to `/tmp/kubespray_cache`. On subsequent provisioning runs, this local cache will be used to provision the nodes, minimizing bandwidth usage and improving provisioning time. Expect about 800MB of disk space to be used on the ansible node for the cache. Disk space required for the image cache on the kubernetes nodes is a much as is needed for the largest image, which is currently slightly less than 150MB. +* By default, if `download_run_once` is false, kubespray will not retrieve the downloaded images and files from the download delegate node to the local cache, or use that cache to pre-provision those nodes. If you have a full cache with container images and files and you don’t need to download anything, but want to use a cache - set `download_force_cache` to `True`. +* By default, cached images that are used to pre-provision the remote nodes will be deleted from the remote nodes after use, to save disk space. Setting `download_keep_remote_cache` will prevent the files from being deleted. This can be useful while developing kubespray, as it can decrease provisioning times. As a consequence, the required storage for images on the remote nodes will increase from 150MB to about 550MB, which is currently the combined size of all required container images. + +Container images and binary files are described by the vars like ``foo_version``, +``foo_download_url``, ``foo_checksum`` for binaries and ``foo_image_repo``, +``foo_image_tag`` or optional ``foo_digest_checksum`` for containers. + +Container images may be defined by its repo and tag, for example: +`andyshinn/dnsmasq:2.72`. Or by repo and tag and sha256 digest: +`andyshinn/dnsmasq@sha256:7c883354f6ea9876d176fe1d30132515478b2859d6fc0cbf9223ffdc09168193`. + +Note, the SHA256 digest and the image tag must be both specified and correspond +to each other. The given example above is represented by the following vars: + +```yaml +dnsmasq_digest_checksum: 7c883354f6ea9876d176fe1d30132515478b2859d6fc0cbf9223ffdc09168193 +dnsmasq_image_repo: andyshinn/dnsmasq +dnsmasq_image_tag: '2.72' +``` + +The full list of available vars may be found in the download's ansible role defaults. Those also allow to specify custom urls and local repositories for binaries and container +images as well. See also the DNS stack docs for the related intranet configuration, +so the hosts can resolve those urls and repos. diff --git a/kubespray/docs/encrypting-secret-data-at-rest.md b/kubespray/docs/encrypting-secret-data-at-rest.md new file mode 100644 index 0000000..f790f15 --- /dev/null +++ b/kubespray/docs/encrypting-secret-data-at-rest.md @@ -0,0 +1,22 @@ +# Encrypting Secret Data at Rest + +Before enabling Encrypting Secret Data at Rest, please read the following documentation carefully. + + + +As you can see from the documentation above, 5 encryption providers are supported as of today (22.02.2022). + +As default value for the provider we have chosen `secretbox`. + +Alternatively you can use the values `identity`, `aesgcm`, `aescbc` or `kms`. + +| Provider | Why we have decided against the value as default | +|----------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| identity | no encryption | +| aesgcm | Must be rotated every 200k writes | +| aescbc | Not recommended due to CBC's vulnerability to padding oracle attacks. | +| kms | Is the official recommended way, but assumes that a key management service independent of Kubernetes exists, we cannot assume this in all environments, so not a suitable default value. | + +## Details about Secretbox + +Secretbox uses [Poly1305](https://cr.yp.to/mac.html) as message-authentication code and [XSalsa20](https://www.xsalsa20.com/) as secret-key authenticated encryption and secret-key encryption. diff --git a/kubespray/docs/equinix-metal.md b/kubespray/docs/equinix-metal.md new file mode 100644 index 0000000..ccdabae --- /dev/null +++ b/kubespray/docs/equinix-metal.md @@ -0,0 +1,100 @@ +# Equinix Metal + +Kubespray provides support for bare metal deployments using the [Equinix Metal](http://metal.equinix.com). +Deploying upon bare metal allows Kubernetes to run at locations where an existing public or private cloud might not exist such +as cell tower, edge collocated installations. The deployment mechanism used by Kubespray for Equinix Metal is similar to that used for +AWS and OpenStack clouds (notably using Terraform to deploy the infrastructure). Terraform uses the Equinix Metal provider plugin +to provision and configure hosts which are then used by the Kubespray Ansible playbooks. The Ansible inventory is generated +dynamically from the Terraform state file. + +## Local Host Configuration + +To perform this installation, you will need a localhost to run Terraform/Ansible (laptop, VM, etc) and an account with Equinix Metal. +In this example, we are provisioning a m1.large CentOS7 OpenStack VM as the localhost for the Kubernetes installation. +You'll need Ansible, Git, and PIP. + +```bash +sudo yum install epel-release +sudo yum install ansible +sudo yum install git +sudo yum install python-pip +``` + +## Playbook SSH Key + +An SSH key is needed by Kubespray/Ansible to run the playbooks. +This key is installed into the bare metal hosts during the Terraform deployment. +You can generate a key new key or use an existing one. + +```bash +ssh-keygen -f ~/.ssh/id_rsa +``` + +## Install Terraform + +Terraform is required to deploy the bare metal infrastructure. The steps below are for installing on CentOS 7. +[More terraform installation options are available.](https://learn.hashicorp.com/terraform/getting-started/install.html) + +Grab the latest version of Terraform and install it. + +```bash +echo "https://releases.hashicorp.com/terraform/$(curl -s https://checkpoint-api.hashicorp.com/v1/check/terraform | jq -r -M '.current_version')/terraform_$(curl -s https://checkpoint-api.hashicorp.com/v1/check/terraform | jq -r -M '.current_version')_linux_amd64.zip" +sudo yum install unzip +sudo unzip terraform_0.14.10_linux_amd64.zip -d /usr/local/bin/ +``` + +## Download Kubespray + +Pull over Kubespray and setup any required libraries. + +```bash +git clone https://github.com/kubernetes-sigs/kubespray +cd kubespray +``` + +## Install Ansible + +Install Ansible according to [Ansible installation guide](/docs/ansible.md#installing-ansible) + +## Cluster Definition + +In this example, a new cluster called "alpha" will be created. + +```bash +cp -LRp contrib/terraform/packet/sample-inventory inventory/alpha +cd inventory/alpha/ +ln -s ../../contrib/terraform/packet/hosts +``` + +Details about the cluster, such as the name, as well as the authentication tokens and project ID +for Equinix Metal need to be defined. To find these values see [Equinix Metal API Accounts](https://metal.equinix.com/developers/docs/accounts/). + +```bash +vi cluster.tfvars +``` + +* cluster_name = alpha +* packet_project_id = ABCDEFGHIJKLMNOPQRSTUVWXYZ123456 +* public_key_path = 12345678-90AB-CDEF-GHIJ-KLMNOPQRSTUV + +## Deploy Bare Metal Hosts + +Initializing Terraform will pull down any necessary plugins/providers. + +```bash +terraform init ../../contrib/terraform/packet/ +``` + +Run Terraform to deploy the hardware. + +```bash +terraform apply -var-file=cluster.tfvars ../../contrib/terraform/packet +``` + +## Run Kubespray Playbooks + +With the bare metal infrastructure deployed, Kubespray can now install Kubernetes and setup the cluster. + +```bash +ansible-playbook --become -i inventory/alpha/hosts cluster.yml +``` diff --git a/kubespray/docs/etcd.md b/kubespray/docs/etcd.md new file mode 100644 index 0000000..2efc85c --- /dev/null +++ b/kubespray/docs/etcd.md @@ -0,0 +1,52 @@ +# etcd + +## Deployment Types + +It is possible to deploy etcd with three methods. To change the default deployment method (host), use the `etcd_deployment_type` variable. Possible values are `host`, `kubeadm`, and `docker`. + +### Host + +Host deployment is the default method. Using this method will result in etcd installed as a systemd service. + +### Docker + +Installs docker in etcd group members and runs etcd on docker containers. Only usable when `container_manager` is set to `docker`. + +### Kubeadm + +This deployment method is experimental and is only available for new deployments. This deploys etcd as a static pod in master hosts. + +## Metrics + +To expose metrics on a separate HTTP port, define it in the inventory with: + +```yaml +etcd_metrics_port: 2381 +``` + +To create a service `etcd-metrics` and associated endpoints in the `kube-system` namespace, +define its labels in the inventory with: + +```yaml +etcd_metrics_service_labels: + k8s-app: etcd + app.kubernetes.io/managed-by: Kubespray + app: kube-prometheus-stack-kube-etcd + release: prometheus-stack +``` + +The last two labels in the above example allows to scrape the metrics from the +[kube-prometheus-stack](https://github.com/prometheus-community/helm-charts/tree/main/charts/kube-prometheus-stack) +chart with the following Helm `values.yaml` : + +```yaml +kubeEtcd: + service: + enabled: false +``` + +To fully override metrics exposition urls, define it in the inventory with: + +```yaml +etcd_listen_metrics_urls: "http://0.0.0.0:2381" +``` diff --git a/kubespray/docs/fcos.md b/kubespray/docs/fcos.md new file mode 100644 index 0000000..c1fe4b7 --- /dev/null +++ b/kubespray/docs/fcos.md @@ -0,0 +1,69 @@ +# Fedora CoreOS + +Tested with stable version 37.20230322.3.0 + +Because package installation with `rpm-ostree` requires a reboot, playbook may fail while bootstrap. +Restart playbook again. + +## Containers + +Tested with + +- containerd +- crio + +## Network + +### calico + +To use calico create sysctl file with ignition: + +```yaml +files: + - path: /etc/sysctl.d/reverse-path-filter.conf + contents: + inline: | + net.ipv4.conf.all.rp_filter=1 +``` + +## libvirt setup + +### Prepare + +Prepare ignition and serve via http (a.e. python -m http.server ) + +```json +{ + "ignition": { + "version": "3.0.0" + }, + + "passwd": { + "users": [ + { + "name": "ansibleUser", + "sshAuthorizedKeys": [ + "ssh-rsa ..publickey.." + ], + "groups": [ "wheel" ] + } + ] + } +} +``` + +### create guest + +```ShellSeasion +machine_name=myfcos1 +ignition_url=http://mywebserver/fcos.ign + +fcos_version=34.20210611.3.0 +kernel=https://builds.coreos.fedoraproject.org/prod/streams/stable/builds/${fcos_version}/x86_64/fedora-coreos-${fcos_version}-live-kernel-x86_64 +initrd=https://builds.coreos.fedoraproject.org/prod/streams/stable/builds/${fcos_version}/x86_64/fedora-coreos-${fcos_version}-live-initramfs.x86_64.img +rootfs=https://builds.coreos.fedoraproject.org/prod/streams/stable/builds/${fcos_version}/x86_64/fedora-coreos-${fcos_version}-live-rootfs.x86_64.img +kernel_args="console=ttyS0 coreos.live.rootfs_url=${rootfs} coreos.inst.install_dev=/dev/sda coreos.inst.stream=stable coreos.inst.ignition_url=${ignition_url}" +sudo virt-install --name ${machine_name} --ram 4048 --graphics=none --vcpus 2 --disk size=20 \ + --network bridge=virbr0 \ + --install kernel=${kernel},initrd=${initrd},kernel_args_overwrite=yes,kernel_args="${kernel_args}" +``` diff --git a/kubespray/docs/figures/kubespray-calico-rr.png b/kubespray/docs/figures/kubespray-calico-rr.png new file mode 100644 index 0000000000000000000000000000000000000000..2dacdb5156d3a963b84d12a55138ba1b8a8680a3 GIT binary patch literal 40710 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 literal 0 HcmV?d00001 diff --git a/kubespray/docs/figures/loadbalancer_localhost.png b/kubespray/docs/figures/loadbalancer_localhost.png new file mode 100644 index 0000000000000000000000000000000000000000..0732d5489a919007e65f58f15ac0e181b612e7ee GIT binary patch literal 58266 zcmeFZWmr{R)HVu;l1hn63cBeM36W0On^d|(x*Mb$1f{#XJEc(&5D<{=?(UF0a~sd| zeDArwpXblH-d`fR)|zX~F-P3@JyyVLDbc&=gy;we2zOynVHpI3Tj2-@Na?6|z;7~L z7BLVI?uo`JZTcq$oKqfSvk9ty5WO|towX)==V9d|btJEnhueC*42lY#D)H5dJu^UwFGZekg zVg>$JNJr`K-EqeFz62&`Gh!VG7kw(>7Q0UWswfhrrhXxyRJyZq;v5j0vwbG7U7)vh zSi*XMarFe^cP)d;;oc$AFm&U8z{7o@SJW}z_v|tTnXrUUDtUOa=90Ph@UkcR?6RnX zZ^d5v=i>t6j)P6ef%gF)b;7V!?N^?$)a%GhpWycokRN_v=(H#w8LXyiBpS9-3i#$L zr|cuuAZ%rgcYGO^Cw)kDOm;B;sL9TV#+4SZu7|VMJbkAVFWa<9`4XxL|aY0 zSfrLn!$H?hZsl2T5B`jYfy2>4i0q{Beey|vmFSFtF=iK!{1eK8XTgUho<{1qc8tR0 z&&bT%s{>BUgapk3)B*=Gi>*du(3BWH=bN5Um=1q@wQtlxyy|c#(tJ4n(9>Wq%dM;Q z8>QKnwea`00C}km_7nRBan|%8XlDq|$~?VsHud#X?sE-M{FVr7e`1|SMc&r5S*Fu| z)7KvZwGA(d_mnem?<+z5JXz5@$uq~U0isN4uNsP5EKEUa1mt0-28`T^ygYot z`&Za;S@=HT;&Eaf!H*jAOiZ|Q1jiZ2v`n53n3;`i-KFQ)b{59U7HV)CDC%cu_BFAwYn z4lfQi#pS-N_1UfXv;V& z_Y3W27&Pib40Df^t#4RU|J^Fi0~x~%J$;H7<-|z^gkj;VagjOJc4xebe$E;pv*x9% zK?d6|X7t$Wn#+3_2jp~V_QC}hbIHl%bIQKTkblX+-z>dXgP>Pi5+wdQ5Ha=pS<3*1 zXO)hg#8!0ndRmdJcF$g$HxcUMEqC5gL9HJsg0Pp-n>jL10&OrH85OHH^3mc}XwOfQ zH>aOtrS@6-*4xrtlC)>T+MUVZ=Qc%`3G9x*q^0r|lamvNlb^CV3F z+KFTT^_5n6eKUui)rCiL49GQcDZFzTZ>Vd=3%QLnE=h#^^ zxhGeb(U12nFP>8zbAA_J?){*>#6fu`efWI2?DCO7_3G&R%q0<4DqX$lg2Ptq5E-&cyW-5$ca6Kj9(U4{u6OO(hJQ zVx(V;LOI@xRv1Klc(!Gg!@x5a@5l)=rz7dN;Xf-uvtr(&eZqHMMDR@O!LtvKuy4JcYfe6I;yoeg zjmz6vPaYh!8MU+OvCOX9D5^Jh+F1XzQAF{8L8bB=6x#yYLdf5nm?&A-{qcncg5+9^CXGG zly*0Lg@iJ7>v3NS->gcd;Tr`*PNs6rN(A2&CGh zkwHv8?HV}}AiKDYS=7wm;W}T-*j`O>w`V)vnV!7f4{;w07iUy{)$;*kTm?T|(`Dbc zsOfy;WHE$Kg_&8CeT$a%vw@nz8=L%rA!D3y7x!|O2C?JYvzuE-_|d>+kHHfNaIoJC zqz++SWW{J0Q>D0At9N!hc5euJ#GIZ?#qz4kE@m!_wSGN2A(2B-5XyM%fk?UR(z>9j z${-lND}Bi;b>1c8`ex&l(#V+_3wXB&6QWNh6U~7DnQ-bD)|1*O!WOn8I~nFl-NgJ+L^s?t}?z z!PKaF)x9S(gge(`kp3VMWdH7)>{$BGpg!j+_oVrc4ccUXsb&<9?U~Z~ORe z8V(=jPhl`M{NS|P7$U>cOlVOfq@~^pIAmae z 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 mtu 1450 qdisc noqueue state UNKNOWN group default + link/ether e2:f3:a7:0f:bf:cb brd ff:ff:ff:ff:ff:ff + inet 10.233.16.0/18 scope global flannel.1 + valid_lft forever preferred_lft forever + inet6 fe80::e0f3:a7ff:fe0f:bfcb/64 scope link + valid_lft forever preferred_lft forever +``` + +* Try to run a container and check its ip address + +```ShellSession +kubectl run test --image=busybox --command -- tail -f /dev/null +replicationcontroller "test" created + +kubectl describe po test-34ozs | grep ^IP +IP: 10.233.16.2 +``` + +```ShellSession +kubectl exec test-34ozs -- ip a show dev eth0 +8: eth0@if9: mtu 1450 qdisc noqueue + link/ether 02:42:0a:e9:2b:03 brd ff:ff:ff:ff:ff:ff + inet 10.233.16.2/24 scope global eth0 + valid_lft forever preferred_lft forever + inet6 fe80::42:aff:fee9:2b03/64 scope link tentative flags 08 + valid_lft forever preferred_lft forever +``` diff --git a/kubespray/docs/flatcar.md b/kubespray/docs/flatcar.md new file mode 100644 index 0000000..cdd2c6a --- /dev/null +++ b/kubespray/docs/flatcar.md @@ -0,0 +1,14 @@ +Flatcar Container Linux bootstrap +=============== + +Example with Ansible: + +Before running the cluster playbook you must satisfy the following requirements: + +General Flatcar Pre-Installation Notes: + +- Ensure that the bin_dir is set to `/opt/bin` +- ansible_python_interpreter should be `/opt/bin/python`. This will be laid down by the bootstrap task. +- The resolvconf_mode setting of `docker_dns` **does not** work for Flatcar. This is because we do not edit the systemd service file for docker on Flatcar nodes. Instead, just use the default `host_resolvconf` mode. It should work out of the box. + +Then you can proceed to [cluster deployment](#run-deployment) diff --git a/kubespray/docs/gcp-lb.md b/kubespray/docs/gcp-lb.md new file mode 100644 index 0000000..8e8f8c4 --- /dev/null +++ b/kubespray/docs/gcp-lb.md @@ -0,0 +1,20 @@ +# GCP Load Balancers for type=LoadBalacer of Kubernetes Services + +Google Cloud Platform can be used for creation of Kubernetes Service Load Balancer. + +This feature is able to deliver by adding parameters to `kube-controller-manager` and `kubelet`. You need specify: + +```ShellSession + --cloud-provider=gce + --cloud-config=/etc/kubernetes/cloud-config +``` + +To get working it in kubespray, you need to add tag to GCE instances and specify it in kubespray group vars and also set `cloud_provider` to `gce`. So for example, in file `group_vars/all/gcp.yml`: + +```yaml + cloud_provider: gce + gce_node_tags: k8s-lb +``` + +When you will setup it and create SVC in Kubernetes with `type=LoadBalancer`, cloud provider will create public IP and will set firewall. +Note: Cloud provider run under VM service account, so this account needs to have correct permissions to be able to create all GCP resources. diff --git a/kubespray/docs/gcp-pd-csi.md b/kubespray/docs/gcp-pd-csi.md new file mode 100644 index 0000000..88fa060 --- /dev/null +++ b/kubespray/docs/gcp-pd-csi.md @@ -0,0 +1,77 @@ +# GCP Persistent Disk CSI Driver + +The GCP Persistent Disk CSI driver allows you to provision volumes for pods with a Kubernetes deployment over Google Cloud Platform. The CSI driver replaces to volume provioning done by the in-tree azure cloud provider which is deprecated. + +To deploy GCP Persistent Disk CSI driver, uncomment the `gcp_pd_csi_enabled` option in `group_vars/all/gcp.yml` and set it to `true`. + +## GCP Persistent Disk Storage Class + +If you want to deploy the GCP Persistent Disk storage class to provision volumes dynamically, you should set `persistent_volumes_enabled` in `group_vars/k8s_cluster/k8s_cluster.yml` to `true`. + +## GCP credentials + +In order for the CSI driver to provision disks, you need to create for it a service account on GCP with the appropriate permissions. + +Follow these steps to configure it: + +```ShellSession +# This will open a web page for you to authenticate +gcloud auth login +export PROJECT=nameofmyproject +gcloud config set project $PROJECT + +git clone https://github.com/kubernetes-sigs/gcp-compute-persistent-disk-csi-driver $GOPATH/src/sigs.k8s.io/gcp-compute-persistent-disk-csi-driver + +export GCE_PD_SA_NAME=my-gce-pd-csi-sa +export GCE_PD_SA_DIR=/my/safe/credentials/directory + +./deploy/setup-project.sh +``` + +The above will create a file named `cloud-sa.json` in the specified `GCE_PD_SA_DIR`. This file contains the service account with the appropriate credentials for the CSI driver to perform actions on GCP to request disks for pods. + +You need to provide this file's path through the variable `gcp_pd_csi_sa_cred_file` in `inventory/mycluster/group_vars/all/gcp.yml` + +You can now deploy Kubernetes with Kubespray over GCP. + +## GCP PD CSI Driver test + +To test the dynamic provisioning using GCP PD CSI driver, make sure to have the storage class deployed (through persistent volumes), and apply the following manifest: + +```yml +--- +kind: PersistentVolumeClaim +apiVersion: v1 +metadata: + name: podpvc +spec: + accessModes: + - ReadWriteOnce + storageClassName: csi-gce-pd + resources: + requests: + storage: 1Gi + +--- +apiVersion: v1 +kind: Pod +metadata: + name: web-server +spec: + containers: + - name: web-server + image: nginx + volumeMounts: + - mountPath: /var/lib/www/html + name: mypvc + volumes: + - name: mypvc + persistentVolumeClaim: + claimName: podpvc + readOnly: false +``` + +## GCP PD documentation + +You can find the official GCP Persistent Disk CSI driver installation documentation here: [GCP PD CSI Driver](https://github.com/kubernetes-sigs/gcp-compute-persistent-disk-csi-driver/blob/master/docs/kubernetes/user-guides/driver-install.md +) diff --git a/kubespray/docs/getting-started.md b/kubespray/docs/getting-started.md new file mode 100644 index 0000000..32660d1 --- /dev/null +++ b/kubespray/docs/getting-started.md @@ -0,0 +1,144 @@ +# Getting started + +## Building your own inventory + +Ansible inventory can be stored in 3 formats: YAML, JSON, or INI-like. There is +an example inventory located +[here](https://github.com/kubernetes-sigs/kubespray/blob/master/inventory/sample/inventory.ini). + +You can use an +[inventory generator](https://github.com/kubernetes-sigs/kubespray/blob/master/contrib/inventory_builder/inventory.py) +to create or modify an Ansible inventory. Currently, it is limited in +functionality and is only used for configuring a basic Kubespray cluster inventory, but it does +support creating inventory file for large clusters as well. It now supports +separated ETCD and Kubernetes control plane roles from node role if the size exceeds a +certain threshold. Run `python3 contrib/inventory_builder/inventory.py help` for more information. + +Example inventory generator usage: + +```ShellSession +cp -r inventory/sample inventory/mycluster +declare -a IPS=(10.10.1.3 10.10.1.4 10.10.1.5) +CONFIG_FILE=inventory/mycluster/hosts.yml python3 contrib/inventory_builder/inventory.py ${IPS[@]} +``` + +Then use `inventory/mycluster/hosts.yml` as inventory file. + +## Starting custom deployment + +Once you have an inventory, you may want to customize deployment data vars +and start the deployment: + +**IMPORTANT**: Edit my\_inventory/groups\_vars/\*.yaml to override data vars: + +```ShellSession +ansible-playbook -i inventory/mycluster/hosts.yml cluster.yml -b -v \ + --private-key=~/.ssh/private_key +``` + +See more details in the [ansible guide](/docs/ansible.md). + +### Adding nodes + +You may want to add worker, control plane or etcd nodes to your existing cluster. This can be done by re-running the `cluster.yml` playbook, or you can target the bare minimum needed to get kubelet installed on the worker and talking to your control planes. This is especially helpful when doing something like autoscaling your clusters. + +- Add the new worker node to your inventory in the appropriate group (or utilize a [dynamic inventory](https://docs.ansible.com/ansible/latest/user_guide/intro_inventory.html)). +- Run the ansible-playbook command, substituting `cluster.yml` for `scale.yml`: + +```ShellSession +ansible-playbook -i inventory/mycluster/hosts.yml scale.yml -b -v \ + --private-key=~/.ssh/private_key +``` + +### Remove nodes + +You may want to remove **control plane**, **worker**, or **etcd** nodes from your +existing cluster. This can be done by re-running the `remove-node.yml` +playbook. First, all specified nodes will be drained, then stop some +kubernetes services and delete some certificates, +and finally execute the kubectl command to delete these nodes. +This can be combined with the add node function. This is generally helpful +when doing something like autoscaling your clusters. Of course, if a node +is not working, you can remove the node and install it again. + +Use `--extra-vars "node=,"` to select the node(s) you want to delete. + +```ShellSession +ansible-playbook -i inventory/mycluster/hosts.yml remove-node.yml -b -v \ +--private-key=~/.ssh/private_key \ +--extra-vars "node=nodename,nodename2" +``` + +If a node is completely unreachable by ssh, add `--extra-vars reset_nodes=false` +to skip the node reset step. If one node is unavailable, but others you wish +to remove are able to connect via SSH, you could set `reset_nodes=false` as a host +var in inventory. + +## Connecting to Kubernetes + +By default, Kubespray configures kube_control_plane hosts with insecure access to +kube-apiserver via port 8080. A kubeconfig file is not necessary in this case, +because kubectl will use to connect. The kubeconfig files +generated will point to localhost (on kube_control_planes) and kube_node hosts will +connect either to a localhost nginx proxy or to a loadbalancer if configured. +More details on this process are in the [HA guide](/docs/ha-mode.md). + +Kubespray permits connecting to the cluster remotely on any IP of any +kube_control_plane host on port 6443 by default. However, this requires +authentication. One can get a kubeconfig from kube_control_plane hosts +(see [below](#accessing-kubernetes-api)). + +For more information on kubeconfig and accessing a Kubernetes cluster, refer to +the Kubernetes [documentation](https://kubernetes.io/docs/tasks/access-application-cluster/configure-access-multiple-clusters/). + +## Accessing Kubernetes Dashboard + +Supported version is kubernetes-dashboard v2.0.x : + +- Login option : token/kubeconfig by default +- Deployed by default in "kube-system" namespace, can be overridden with `dashboard_namespace: kubernetes-dashboard` in inventory, +- Only serves over https + +Access is described in [dashboard docs](https://github.com/kubernetes/dashboard/tree/master/docs/user/accessing-dashboard). With kubespray's default deployment in kube-system namespace, instead of kubernetes-dashboard : + +- Proxy URL is +- kubectl commands must be run with "-n kube-system" + +Accessing through Ingress is highly recommended. For proxy access, please note that proxy must listen to [localhost](https://github.com/kubernetes/dashboard/issues/692#issuecomment-220492484) (`proxy --address="x.x.x.x"` will not work) + +For token authentication, guide to create Service Account is provided in [dashboard sample user](https://github.com/kubernetes/dashboard/blob/master/docs/user/access-control/creating-sample-user.md) doc. Still take care of default namespace. + +Access can also by achieved via ssh tunnel on a control plane : + +```bash +# localhost:8081 will be sent to control-plane-1's own localhost:8081 +ssh -L8001:localhost:8001 user@control-plane-1 +sudo -i +kubectl proxy +``` + +## Accessing Kubernetes API + +The main client of Kubernetes is `kubectl`. It is installed on each kube_control_plane +host and can optionally be configured on your ansible host by setting +`kubectl_localhost: true` and `kubeconfig_localhost: true` in the configuration: + +- If `kubectl_localhost` enabled, `kubectl` will download onto `/usr/local/bin/` and setup with bash completion. A helper script `inventory/mycluster/artifacts/kubectl.sh` also created for setup with below `admin.conf`. +- If `kubeconfig_localhost` enabled `admin.conf` will appear in the `inventory/mycluster/artifacts/` directory after deployment. +- The location where these files are downloaded to can be configured via the `artifacts_dir` variable. + +NOTE: The controller host name in the admin.conf file might be a private IP. If so, change it to use the controller's public IP or the cluster's load balancer. + +You can see a list of nodes by running the following commands: + +```ShellSession +cd inventory/mycluster/artifacts +./kubectl.sh get nodes +``` + +If desired, copy admin.conf to ~/.kube/config. + +## Setting up your first cluster + +[Setting up your first cluster](/docs/setting-up-your-first-cluster.md) is an + applied step-by-step guide for setting up your first cluster with Kubespray. diff --git a/kubespray/docs/gvisor.md b/kubespray/docs/gvisor.md new file mode 100644 index 0000000..ef0a64b --- /dev/null +++ b/kubespray/docs/gvisor.md @@ -0,0 +1,16 @@ +# gVisor + +[gVisor](https://gvisor.dev/docs/) is an application kernel, written in Go, that implements a substantial portion of the Linux system call interface. It provides an additional layer of isolation between running applications and the host operating system. + +gVisor includes an Open Container Initiative (OCI) runtime called runsc that makes it easy to work with existing container tooling. The runsc runtime integrates with Docker and Kubernetes, making it simple to run sandboxed containers. + +## Usage + +To enable gVisor you should be using a container manager that is compatible with selecting the [RuntimeClass](https://kubernetes.io/docs/concepts/containers/runtime-class/) such as `containerd`. + +Containerd support: + +```yaml +container_manager: containerd +gvisor_enabled: true +``` diff --git a/kubespray/docs/ha-mode.md b/kubespray/docs/ha-mode.md new file mode 100644 index 0000000..f961c74 --- /dev/null +++ b/kubespray/docs/ha-mode.md @@ -0,0 +1,158 @@ +# HA endpoints for K8s + +The following components require a highly available endpoints: + +* etcd cluster, +* kube-apiserver service instances. + +The latter relies on a 3rd side reverse proxy, like Nginx or HAProxy, to +achieve the same goal. + +## Etcd + +The etcd clients (kube-api-masters) are configured with the list of all etcd peers. If the etcd-cluster has multiple instances, it's configured in HA already. + +## Kube-apiserver + +K8s components require a loadbalancer to access the apiservers via a reverse +proxy. Kubespray includes support for an nginx-based proxy that resides on each +non-master Kubernetes node. This is referred to as localhost loadbalancing. It +is less efficient than a dedicated load balancer because it creates extra +health checks on the Kubernetes apiserver, but is more practical for scenarios +where an external LB or virtual IP management is inconvenient. This option is +configured by the variable `loadbalancer_apiserver_localhost` (defaults to +`True`. Or `False`, if there is an external `loadbalancer_apiserver` defined). +You may also define the port the local internal loadbalancer uses by changing, +`loadbalancer_apiserver_port`. This defaults to the value of +`kube_apiserver_port`. It is also important to note that Kubespray will only +configure kubelet and kube-proxy on non-master nodes to use the local internal +loadbalancer. If you wish to control the name of the loadbalancer container, +you can set the variable `loadbalancer_apiserver_pod_name`. + +If you choose to NOT use the local internal loadbalancer, you will need to +use the [kube-vip](kube-vip.md) ansible role or configure your own loadbalancer to achieve HA. By default, it only configures a non-HA endpoint, which points to the +`access_ip` or IP address of the first server node in the `kube_control_plane` group. +It can also configure clients to use endpoints for a given loadbalancer type. +The following diagram shows how traffic to the apiserver is directed. + +![Image](figures/loadbalancer_localhost.png?raw=true) + +A user may opt to use an external loadbalancer (LB) instead. An external LB +provides access for external clients, while the internal LB accepts client +connections only to the localhost. +Given a frontend `VIP` address and `IP1, IP2` addresses of backends, here is +an example configuration for a HAProxy service acting as an external LB: + +```raw +listen kubernetes-apiserver-https + bind :8383 + mode tcp + option log-health-checks + timeout client 3h + timeout server 3h + server master1 :6443 check check-ssl verify none inter 10000 + server master2 :6443 check check-ssl verify none inter 10000 + balance roundrobin +``` + + Note: That's an example config managed elsewhere outside Kubespray. + +And the corresponding example global vars for such a "cluster-aware" +external LB with the cluster API access modes configured in Kubespray: + +```yml +apiserver_loadbalancer_domain_name: "my-apiserver-lb.example.com" +loadbalancer_apiserver: + address: + port: 8383 +``` + + Note: The default kubernetes apiserver configuration binds to all interfaces, + so you will need to use a different port for the vip from that the API is + listening on, or set the `kube_apiserver_bind_address` so that the API only + listens on a specific interface (to avoid conflict with haproxy binding the + port on the VIP address) + +This domain name, or default "lb-apiserver.kubernetes.local", will be inserted +into the `/etc/hosts` file of all servers in the `k8s_cluster` group and wired +into the generated self-signed TLS/SSL certificates as well. Note that +the HAProxy service should as well be HA and requires a VIP management, which +is out of scope of this doc. + +There is a special case for an internal and an externally configured (not with +Kubespray) LB used simultaneously. Keep in mind that the cluster is not aware +of such an external LB and you need no to specify any configuration variables +for it. + + Note: TLS/SSL termination for externally accessed API endpoints' will **not** + be covered by Kubespray for that case. Make sure your external LB provides it. + Alternatively you may specify an external load balanced VIPs in the + `supplementary_addresses_in_ssl_keys` list. Then, kubespray will add them into + the generated cluster certificates as well. + +Aside of that specific case, the `loadbalancer_apiserver` considered mutually +exclusive to `loadbalancer_apiserver_localhost`. + +Access API endpoints are evaluated automatically, as the following: + +| Endpoint type | kube_control_plane | non-master | external | +|------------------------------|------------------------------------------|-------------------------|-----------------------| +| Local LB (default) | `https://dbip:sp` | `https://lc:nsp` | `https://m[0].aip:sp` | +| Local LB (default) + cbip | `https://cbip:sp` and `https://lc:nsp` | `https://lc:nsp` | `https://m[0].aip:sp` | +| Local LB + Unmanaged here LB | `https://dbip:sp` | `https://lc:nsp` | `https://ext` | +| External LB, no internal | `https://dbip:sp` | `` | `https://lb:lp` | +| No ext/int LB | `https://dbip:sp` | `` | `https://m[0].aip:sp` | + +Where: + +* `m[0]` - the first node in the `kube_control_plane` group; +* `lb` - LB FQDN, `apiserver_loadbalancer_domain_name`; +* `ext` - Externally load balanced VIP:port and FQDN, not managed by Kubespray; +* `lc` - localhost; +* `cbip` - a custom bind IP, `kube_apiserver_bind_address`; +* `dbip` - localhost for the default bind IP '0.0.0.0'; +* `nsp` - nginx secure port, `loadbalancer_apiserver_port`, defers to `sp`; +* `sp` - secure port, `kube_apiserver_port`; +* `lp` - LB port, `loadbalancer_apiserver.port`, defers to the secure port; +* `ip` - the node IP, defers to the ansible IP; +* `aip` - `access_ip`, defers to the ip. + +A second and a third column represent internal cluster access modes. The last +column illustrates an example URI to access the cluster APIs externally. +Kubespray has nothing to do with it, this is informational only. + +As you can see, the masters' internal API endpoints are always +contacted via the local bind IP, which is `https://bip:sp`. + +## Optional configurations + +### ETCD with a LB + +In order to use an external loadbalancing (L4/TCP or L7 w/ SSL Passthrough VIP), the following variables need to be overridden in group_vars + +* `etcd_access_addresses` +* `etcd_client_url` +* `etcd_cert_alt_names` +* `etcd_cert_alt_ips` + +#### Example of a VIP w/ FQDN + +```yaml +etcd_access_addresses: https://etcd.example.com:2379 +etcd_client_url: https://etcd.example.com:2379 +etcd_cert_alt_names: + - "etcd.kube-system.svc.{{ dns_domain }}" + - "etcd.kube-system.svc" + - "etcd.kube-system" + - "etcd" + - "etcd.example.com" # This one needs to be added to the default etcd_cert_alt_names +``` + +#### Example of a VIP w/o FQDN (IP only) + +```yaml +etcd_access_addresses: https://2.3.7.9:2379 +etcd_client_url: https://2.3.7.9:2379 +etcd_cert_alt_ips: + - "2.3.7.9" +``` diff --git a/kubespray/docs/hardening.md b/kubespray/docs/hardening.md new file mode 100644 index 0000000..b485c03 --- /dev/null +++ b/kubespray/docs/hardening.md @@ -0,0 +1,137 @@ +# Cluster Hardening + +If you want to improve the security on your cluster and make it compliant with the [CIS Benchmarks](https://learn.cisecurity.org/benchmarks), here you can find a configuration to harden your **kubernetes** installation. + +To apply the hardening configuration, create a file (eg. `hardening.yaml`) and paste the content of the following code snippet into that. + +## Minimum Requirements + +The **kubernetes** version should be at least `v1.23.6` to have all the most recent security features (eg. the new `PodSecurity` admission plugin, etc). + +**N.B.** Some of these configurations have just been added to **kubespray**, so ensure that you have the latest version to make it works properly. Also, ensure that other configurations doesn't override these. + +`hardening.yaml`: + +```yaml +# Hardening +--- + +## kube-apiserver +authorization_modes: ['Node', 'RBAC'] +# AppArmor-based OS +# kube_apiserver_feature_gates: ['AppArmor=true'] +kube_apiserver_request_timeout: 120s +kube_apiserver_service_account_lookup: true + +# enable kubernetes audit +kubernetes_audit: true +audit_log_path: "/var/log/kube-apiserver-log.json" +audit_log_maxage: 30 +audit_log_maxbackups: 10 +audit_log_maxsize: 100 + +tls_min_version: VersionTLS12 +tls_cipher_suites: + - TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 + - TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 + - TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305 + +# enable encryption at rest +kube_encrypt_secret_data: true +kube_encryption_resources: [secrets] +kube_encryption_algorithm: "secretbox" + +kube_apiserver_enable_admission_plugins: + - EventRateLimit + - AlwaysPullImages + - ServiceAccount + - NamespaceLifecycle + - NodeRestriction + - LimitRanger + - ResourceQuota + - MutatingAdmissionWebhook + - ValidatingAdmissionWebhook + - PodNodeSelector + - PodSecurity +kube_apiserver_admission_control_config_file: true +# EventRateLimit plugin configuration +kube_apiserver_admission_event_rate_limits: + limit_1: + type: Namespace + qps: 50 + burst: 100 + cache_size: 2000 + limit_2: + type: User + qps: 50 + burst: 100 +kube_profiling: false + +## kube-controller-manager +kube_controller_manager_bind_address: 127.0.0.1 +kube_controller_terminated_pod_gc_threshold: 50 +# AppArmor-based OS +# kube_controller_feature_gates: ["RotateKubeletServerCertificate=true", "AppArmor=true"] +kube_controller_feature_gates: ["RotateKubeletServerCertificate=true"] + +## kube-scheduler +kube_scheduler_bind_address: 127.0.0.1 +# AppArmor-based OS +# kube_scheduler_feature_gates: ["AppArmor=true"] + +## etcd +etcd_deployment_type: kubeadm + +## kubelet +kubelet_authorization_mode_webhook: true +kubelet_authentication_token_webhook: true +kube_read_only_port: 0 +kubelet_rotate_server_certificates: true +kubelet_protect_kernel_defaults: true +kubelet_event_record_qps: 1 +kubelet_rotate_certificates: true +kubelet_streaming_connection_idle_timeout: "5m" +kubelet_make_iptables_util_chains: true +kubelet_feature_gates: ["RotateKubeletServerCertificate=true", "SeccompDefault=true"] +kubelet_seccomp_default: true +kubelet_systemd_hardening: true +# In case you have multiple interfaces in your +# control plane nodes and you want to specify the right +# IP addresses, kubelet_secure_addresses allows you +# to specify the IP from which the kubelet +# will receive the packets. +kubelet_secure_addresses: "192.168.10.110 192.168.10.111 192.168.10.112" + +# additional configurations +kube_owner: root +kube_cert_group: root + +# create a default Pod Security Configuration and deny running of insecure pods +# kube_system namespace is exempted by default +kube_pod_security_use_default: true +kube_pod_security_default_enforce: restricted +``` + +Let's take a deep look to the resultant **kubernetes** configuration: + +* The `anonymous-auth` (on `kube-apiserver`) is set to `true` by default. This is fine, because it is considered safe if you enable `RBAC` for the `authorization-mode`. +* The `enable-admission-plugins` has not the `PodSecurityPolicy` admission plugin. This because it is going to be definitely removed from **kubernetes** `v1.25`. For this reason we decided to set the newest `PodSecurity` (for more details, please take a look here: ). Then, we set the `EventRateLimit` plugin, providing additional configuration files (that are automatically created under the hood and mounted inside the `kube-apiserver` container) to make it work. +* The `encryption-provider-config` provide encryption at rest. This means that the `kube-apiserver` encrypt data that is going to be stored before they reach `etcd`. So the data is completely unreadable from `etcd` (in case an attacker is able to exploit this). +* The `rotateCertificates` in `KubeletConfiguration` is set to `true` along with `serverTLSBootstrap`. This could be used in alternative to `tlsCertFile` and `tlsPrivateKeyFile` parameters. Additionally it automatically generates certificates by itself. By default the CSRs are approved automatically via [kubelet-csr-approver](https://github.com/postfinance/kubelet-csr-approver). You can customize approval configuration by modifying Helm values via `kubelet_csr_approver_values`. + See for more information on the subject. +* If you are installing **kubernetes** in an AppArmor-based OS (eg. Debian/Ubuntu) you can enable the `AppArmor` feature gate uncommenting the lines with the comment `# AppArmor-based OS` on top. +* The `kubelet_systemd_hardening`, both with `kubelet_secure_addresses` setup a minimal firewall on the system. To better understand how these variables work, here's an explanatory image: + ![kubelet hardening](img/kubelet-hardening.png) + +Once you have the file properly filled, you can run the **Ansible** command to start the installation: + +```bash +ansible-playbook -v cluster.yml \ + -i inventory.ini \ + -b --become-user=root \ + --private-key ~/.ssh/id_ecdsa \ + -e "@vars.yaml" \ + -e "@hardening.yaml" +``` + +**N.B.** The `vars.yaml` contains our general cluster information (SANs, load balancer, dns, etc..) and `hardening.yaml` is the file described above. diff --git a/kubespray/docs/img/kubelet-hardening.png b/kubespray/docs/img/kubelet-hardening.png new file mode 100644 index 0000000000000000000000000000000000000000..5546a8ba9069cd681b3d13d7fffdb9a3a59c8418 GIT binary patch literal 1547778 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

iLf_mOgy?8e#br1W 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�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ℜfK=bR9fPi#z#Imt%>h`Yl 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


Ty9Uc6#8YwZ$OG--Ojz&bAc#abW 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Ȕpa{=3vZU6qqnJ4frkFIy!Tlq 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≻^rzWXQVF^Y29!a*~b5# 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
  • Wnzt=QNSw+RreT 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�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

    DM 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 AR7vvPj3HY|_f5Gd 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%0ACWTNj|h5a}66I4pmWvj9jd^jy~_o8tS!V2ghK zW`or%<<)@&<>g-hs=pYm22j4dc@>;n_3nt${pUB;ZjP_1(KQj(-h7%EQI5fgz!IFH zW#QR;^z9UrWP@t~5`L%rYip+8rf|Q)(`#taJy#tZma=wYkChM9$ zWV~o*(f-z4H^4+XE?0z<{Uy>jbE0|danYJfNzk9D(AW1QOessiX`)c>+?=+M3PAr4 z#KxOO0=NUh!)|qtufExb!oUnOAojSYDaD=HAKcJg6_5m7dQeMG;Gmh_z$RLmT$C$D z@4kTw^5tEVK+WR;-(vJvMQ9OK#P3J7hnw{Gg`!0qT9m)ps?;=`7YlW(M}za1OP`F1 zk>bp)X2?IM&|Dp#X{|A~9%a)3D#l&8s8qBAZhc2;t3kFEMnexm&7>T@P@n|BWX`1o zVicB5)zD!4G%d3i3e2k365|1{YcDP2jSpDZz)%d&+-!}t`f3F_Q)NQUv*CGX`#dWM z30N|ccn(Z#V0SU&9n{}pc)j>TiaciDna7Deq6P zHzkS?h;*$D7@^ai^};Hyx%4;7TYmRmE(_YfDmo#eK`XSEx@CzDX0ADUX;l|2?OmKe zW1AM^>f5(($`m#0Yb6$kCEoCNb@b#FAP>UFZ00(0UlJ0e@Sf9~04g?sgnN0NHJ?~L zc~XA&^(-D>(FttBCiZ08EU{5S#YWa+WdrkVv$p*=HmvtbpB{{onjO2V7O82A>T{u;08~)jxo5DU$#zGaShI>L6zxbki<9 zPJ7r`i#ydgu~sA3y{1QF(Kt;!VT#_bCzzc;5B&dm z>l*F|>>f}ocnGRH`aiNXYl?7%Bov{^W!@kUKQm0@-Gjb6=Ul4xPGS*e87RFJ01c`> zN5zZ9Jpz<(FOw%&Lq}X3h1*+EXr<+D`NmriEEYB zfXh;eiGcoS(;;pp4fl7zdrOC63<&1rIsi=I8u1=-o^SnsAKWGM{JRfiUSJ|c6Y_-tm1`VYD`|1Z{M+Z2Cqh4s3X28oi|@y=k`|)oqGGY zc9(qn%>mUh=APD{${;*y@McZW?%hlWm0vA7%c)&LCWb|5i(UI&%IqojtXB)R%*d@X zYEhT`+jK0Lo}rNgeLBbBy$%=`nYvGg==&@r$qcLGtN*`S*~7KC^{gqf4W9nxlnelC z3mb-`O8R}`FB=$sB|pALZ`t;!f&X0RTS@exK7=GhvI>-jrx;0}s4Z%GMMaAxFERA> z#fuk%4Fleiwm!MX;Sa1yXMw3Eg_bZ9S!rpd*2%9#P+~H&?`SR)kJL=9@c~_3?3XVR zx*n;obYC@yiy{>A?)oSkgzCTcC(A@&wI?K;yx8*eoAH#gk)6;;7MOV6@h2ezq)fM? z-iKdx9le<|h}?-``JydPwR5nA-kEnd$0l}hg>!RSz0_b!g9$#Mwzuqt4YpW^%1A(| zJG#SV-n@S}4v6t@OPCZBF`&2wfO>k)IM%;+5K`gatEmQ_sLuI?;lW?>gq@T>Rr;u~ z&a%xL6cW0cXgHw1;KcU+?IXNblhDkV-v5kAMZzt`GtII0F6tJwQp+O_SMZ?D427-f z7ZrEy>bS0+#ZN37oS~E{M)SKs+kpm#${7j9pP}14*@HyUBX@PSn3UW$+FinTj_xx&kS z=zXRDWGZFTAW&Im(iyoxsl{!ZG!he;JL94xFnUHRq{mN32n3jyH}T@>2Ut8q1Lw#~ z2T!bKY&IK`8lW!)Z@PA+%m#BIz&K#0Ywgb8;6BydEdZMF-+HVophg9#x&*!E_2{Ld z?XolY4Wwl4GImE3B-@lGUuCYCPSkq%t8q~)(NG3IB;ZKX-U zIH%Xv_~8*khs#Xnj{QzwqOO-p-^akn0>)&L2zzj;s;ON&O1s<+S%+N#@b3^-(1RFB z?+Y!kL^%8zQc!X{&vnRaxP#(xF~+IzDOl zFE49PT`->1kKK<5)R(yzU+chc+dFT$rtnRD-zb#1(wYGRDB3#&SHH?MkE&sM%Oy^0 zLn-S6n|v${o51ep*bhiD%HZgyDfG-_i}!57mCeei^}!C)nfoVtOP5=vMnXpq7MrGa z>>r0WA-wymOJ~<0-XWhwbnFvU+y<*yur8E+-WNtZB5n45?67H&rKp*LOz&6>vOm`) zIE$Lg#N;}PT)Fx52KB(G7T_be@K*l){s82I#$%xRR;AvUb3!eIdMz9!_^!Rw>t5&< zUPst@D-fIBsHr!dB#&ZItOsg81B0{}N3d79KiFJs)5qF}#y3^4vzf6IJrmU*>&MH@ z&mYk=vd#5eA^-Z+V^+U3=n>(ZieykyTS477Wn#(eDT$-BE!KE*@qm~Dk8kB89wnvX zHL7y&XkQv}t-~{30krCGU|w5JPjnk0mA#&A&Ig;k;fNl${vSQ@)YaFf`9~N};nh3T zmIWm)-3H}`Z+t3O!ppjyaVaQ3URB)R`w2)F8>0jIbB2qH;$SV3=iq1lK55x>DnB%! z=+FsHJu_>^ub|H*B|!BtQsdJLi~;zZT<+k|(jzQtH^yP4gIoGWY?G1|m~-<=W8=@H z%>O>a0)m}oVcV56To}bSZKR{r=^~N~v~6eIu6Un2j7a|XoYbm2YP%>8XT~Of@lgPd zuMjb8S3HBfoO3t$NFOj^Hja5x#W6uu(@)hRuK}Vb`!Tk5kH$9k)^oXBKuV6`6*I>x zg+L=fr3h|oed-e5vkJK(f)to^Nv4rU_{b$)^l zOL^iz5J_TnjA0wS2YYuhe#GE5n*kpEv82z?Wvw_XHTlyZ*I!nKP!UMO3we6hx&aT0 zV}61Ty&v#_g~`mesU_E4+Zx$tnYDXuyU-1Ng4 z(-h*P;J-$wovV#$Or4*HNa!w3c2u4p%=A-cUg2A@^+s6@6BBLmp7)1rhPu_|cg7|y zwQmMZ{G(9b8~U|@Rr;|&9Zkv8*7{j_bI37 z^tnzvW~E{4WU+&iU%)dgIHD(lN$iFnViOVs(|`tQT66WmffJ%k3`npw1h}r|voB;s ze6zd`HdGZl-#$YpjFnGw{f~^ z^44PrzCI1ciHLj3Gs|*3d#r8N0T_MxV3wJ1$^9!j*orQ*Ju+k(chOQoP?!1;~*PS0A9bUS-ihiUs>0IR4|tRNM*FvbqMQs?U|m zUgN(!Ha+vlo@#}Qgh{GjZ$v?CD6&#)>@M?b(g3H7B zF?QaE8&(TU))B&*vsd|Q%IHY_VJN+&GeP{6rW<^564u&$G3~MHcQsZ%S9d#Qnvg@u z%CqMY?<#!Oxzx`95N3oM&)S;#2to-4U6bSQSUQh37Aal~Df+oQnnjX_Ij4zSiuE|U z=uQ=z97ni~FjEccO*=QDr-3U|!AEs3Jlx>NP9wgnKe|El5&bFXKl%p-Pt%-sU=0#$ z0#Yy0_l1fgGq$_3`3bJ~=QOjSY6{$+xAw_0#{Iw7zWnEQog-kYXgCw~`#aC`y|Z}_ zwsnEQlOs)KjcRKBqMFGhF4FJ4vcuBKH5MLyX>+lihwN0hNV%{!+DjJ3PA_f>3z5qK zW?Q~2jjF1ucZLQ8b4M1_Sq?~V?XNBg^|f5-a=y8~-@OnguFDIU0}cY(_mcWVuea#{ zdoyY1=mc7<0c@6zv-K+S;nP>rfQj>~&!%bNY=t5ZYX(CDdU%3;LgU=4Z5x5FWonn_ z%_m4Y@cwdP^qoabF}-bZi{~^*%R(QS2C3AiIsIVI&vi|Q>*P;)Q;*VL+tl)0FEm`I zCStKy`n)lIR`$pgT(~zMg>!%z{E6359Og#;UJ~CzmaXeaM}v z2A38qd0aW1%;v2zE4T5F<5z~>fLDi_wlYKRWKGQIY=)fmmk973m7JZJQ1oq_w)9l> zU~D>R=R}4Nm1iVuAm5NE;{V*JK z*T#JYb^r3~p(D%rX9@$^_`T6+)+Y(6$ zB{VR}j**A->zmXi(yw2a)M1zOBrDe=^ta~*?ed3eF>~{SLxw)EmHyhTFgcDt)*5KX zl$-rh5*~ScF2PL0B3RE)#7a*u1uRrGnO=Eyx@8OQG~-Td7|(Z=U3wQo zj#oH~7v<9ge%kMU{W&@+eRYXUW34uWAtQKX2D zJYHQS@T=|(-`!Mo5#5@rfgS9QOcS-mNV1LwWDHtv2fsLXZ;|Mnyz8A5tyu-T%)`;! z=vtzn@XVvQdWY_TytlRVAs2U|l|vi1K)A`OAmNO`o*QGX+zx z*BSM#J&%!Z&!;D3W>wFocC{|g+y&*Jzn)P&8_aOj!yCc|&gY{WFy4g+JS{YD{m_@8 zRQ$hB?9VCsVglAT+czDUbz&=rq>bbwl9%ru+3NU|jJ-C)zTKfmU1-*K|I8y%X?w@w z(N{AH3$q#O?Z)vTOx-iOF&;h^Ha5kTz7+ZttCXc*6WYc|f_iALsPzwZk8Vm^vS&>V z#|rdfEW*Om7>M%nTl|;=6}8sb{^kVt;**lrdW~94ND~GeBjS&lT%Bw+sniyx z6P;BDx{~ut99nzd;~cqcPF7Kbq{yFj1nmu&r8^v1pM#^Cqhl!IzeXP|WKY93ydh57)b2I|@h1FPxfCC0` zt}i0hw>#a=$G{TQL)zw*hxfGo2b`8ce&4H8YE(smZNd1(O_rw~ICfHEysM80=$36N zqN+=r?RLP2yU=O!f|$kkB|%!p=4*1+)$qprND;SYPS5594)`}}v1S-4Z{1nDXEjNjl;0!md-ndl78z5- zk17WA$Q4AqRs@ynUCJsJVmZyan(Y$;PkudhbXR2~=&t^Wsm*)5j-C;t4HBlOO0g!L z1Eygu1{TPapw(VBvg|b-hKmfiFU<>^*tAfc?jXV10p)7nHRjPGYf*pZDWox0M~GDc z11K1nm}1%@1k^pSMKs5n)W}?(=xz%?*J~VCWda0|%G%oNs^e+vI#5~~xrL98#*WK6 z*p{i^#lw$W1*bA=J}t<@QuVYl zb_{(~+>J=6Z^7aV{Wp(t)=W=u14O@%l+ zkNx~GtSrbs*KCI`n89Y1Hq`FHjJuT=Z`2P1A1%aYm?Pg5RovndL?;ib#u`J^?VFXL zr8S+dfR9r0vrJF~a#1LcRMT2M{1ULlKKt=fYoMm=i_t+MUh!fbUv)#XByaHcK@^#H zb!EU)il5N@GXa#}%RGe$I)^C&!S21U(El2F(CP@NAjMWsBknx{Zq0f|*r!jYYB_g_ zihaDYgLl5WswqUhIZhQ5#9`vbDl9Y%sqmYT=_=o9lO^WM^c5p8Hq_)a9Y?uOkP*~W=&Z1Ch0w+0Nu*E=}_9RYKiAf;j=yLfj` zPeV#`xE@b+-_P^G7sNdFAOM~JR8WF-%HYTiOM6DQ?wC_Hy?v&w#u5u)`NHqtV*zKh zX8Dh_+1tLgP07h5hzqR~cbi4W8Z|Mc2gl?@=E zGlbTW)Z|RH@T5IxEKQ5FfjrRXH{7ax_zZT8VY~R{dS^5KNj<9on3Ui$w*QocMNyxl zD)Q)5!^!~NsygM~e${*Zt?EXWyR$#R`s>*vDQA#rX>f7c;?@W>&v|(EL))(@-PjbU zCY6peAyn_%fT(>lN87=oiI!9izaqSi`Br^}cY5JK(|Ajb%E9g(z23DF)&Qr>zBDYC z5|QKEzVZaEiyB0;Bf8%QY=Rp!dqC?)$-07~Lv?CJMSjqC!g;58+E>+GX*!Aodu~;} z`9@qfAD9=d{D8<6=nb>$jD=_M|1XP0SAk2zQa9lHVpGI=ud$}4P2@B|Zxp&c2~k!P z4vL9nxyu+0gBxZ+j_!zESjK|K6xhoIKNLF?SfVbbo=4V3Y&a~_$_~f`x@+fkqwE15>dq& zLAuyGfEC085+vpBlcT%;;t1)bF0OQRKTXjRsbUo{rpAJ4Uwx~p#60|`z^nhm*mnjr zxh-7_f}jyWQBdiMAYD4e@pgd*4|N>S+oA|-SZdJ*Z;A(T+16PiHi z<=YWG=N=FDzTZEBN$}Zw_RN|!Yi4uF;~pcmEM_Y)<}B@EV#1i4O&K`-s7##gyzl#h z>`=@t#r8FBUu44SokovHOSQi8*Bz8|Y>kyM_dnP6wLCFIr}}vGZmu^n+*ng}-{z;B zT%H`cWtY_=dT#NIdq$dP%T8g;Jbj}#d61(Xk>&JNN|PtXz3vtXdECA&60|GxHN4o& zl>qhP8x6&W%Mu2-6OpB>WbTVD3(t~h3CV^RB{8(GJ9@!!)|E;u{AZy1YmxpL^ev77 zK*V<1-_O6I#{@^V8W{58?DKP|l?`Vwp@7r=AV%vs$(bW?@Vc%M2-+#Rc}N zlBK1x^CbxzCF{imP^_X74g@o(6Xr~k+lv?Z_q64fR+a`Vl-^(ERGcsOh@(H1OdI+v zA9M$ax<_zBzmt3glWzk2?0aovo&^<_l8d(zPkmB~<@H}x3a?V+aQL!F0K~INE%I(R zbJ%bBea+r0&D^*n&3^wKvwQ?^7&O;VPFFWx2A3R@DB@VpxljXvd$K4Op4Ln2V znwg%V(PmqFVKn_ZBd3^r>&n!Q{7nAzkD7JQ+2KvyXs+w;eQE<<4*+vvdD>-fe~61h zP;5FrC43-QpwVa7aw^*h?mjOadzdm`wL5?K$`+JGZO&4(Z5$R&+AJQ|;)#As+SEC@ zz4PL|cE{B@H@i;8Zg$|Q5)6g7U==TtGu6yk>`_CjeOKJAG|5^7W=2h53uL&j3|E3k z*Px3!GELso+~1maHG@`Gp9~C_=Y2n1&?X{^G7>U(LA~2rSu|m=JG#;zgA7doH&r91 zPS&*}0SO{L85#W-iL(5_&GWYw;Lnpt@(|n(%l!1(6DyK3mnwF>wfZtooY_%xpQ>N5 z7yl)&5~_E-sa;K*D~&1kGKD=(9c@lzOp{6zfYI^-V=vyErlpk+I=S&^>81={*2cK0 z;#q3;@|{7c=eRQ$1+~D9uGH;M9z0+M6a%u-9B~4dQz-N5>`1!qSfYQ@W>RFLoVb6q z3imDkjH4vl6}5Kmids%1emTZ%7uaXbs$ZO8Y=I-3ySln;8oHZ-t@=aHsH=)w zN3C6Kr$%H^0Y}ZPNd5BmLFkcLpY+Id1GibzO`;P&biZs+ecfQ1v)UL1&WOJNj}gR3_Fu-?QF#|1hjl$Sm-4Oe+hSL@=|a8m}j{k&6a>$Svsm6T$0&(D3IJElLK;ET4l{-;;@Pn)1RM5JvQJT-F!ZKi4K zd+EHd}@+Ee|Z#@7`2+3ld%;-T8w%j@Zp2(`byz2kb-~+eyzCp*oLNsTVsSYrDPu zel)$iS~ZO}z&Z3nRp}O67A#{r)yuAC+-sJ=;S}Ry(wZWLlt4Hoo|!n`Pkfl?erp%G z#qw+Sup{#GgE{Tz5>iK~$qnqxFNnGLL5J89goxuWwsx&8WP~lbyG=!1Q}I3^ zA8C;dO7N$(HeKY7U`yg^8-M5cauW4nmFPuUi~F0YjJ7eD zNizD#bSA!z9G&#Q5AL2GDHubc3G20BH}O{QWS1grrO)Vje2=8@851bgd$Bni@u_UZ z6Wi}49FxTkJ3D>j6!$O_bKgx3(wvx3OIRr=H-YvHqJlKRTasntvO-m`jA~X*GF#F zwn<{r_@^mUo~dh}2Desch~%>Tmq0{*`?*(UdFR~B5uuw*AIEMUt2Q^(D3{2T33}&T zESPy3Vg)ztVafjSoL|%^e&t*g%?tU!DFLKwe%(e#q+Z*OnRFiOYQ9;n*u15RFy!Y4(EG#Ktu(e7b zJnztP=lN1~pWmBU%QYH7G^i63kkic_?>4g*8Ky{%*lq!`+LQegk(&Sv>+is;kv_*o z1@~fjEL(CVMH5)4rinG(_vZT~hiYXu!zy+ULc~;7N^xjLl{ow@kuWnQGC7q#=n!$`sk3p^&_g}JifnnoiI98;d#i6(+_yQ9-wwy@ z$x7~qg%7l%g$~h)78$s?CTT(@mh#{Glzva~%PL(M(aGj<%XjT+~9R*9cC z5r`XZa-Iis6^GDAsoCnqFr49LLQ&zf0nc0Jl^dusAoo&AIzGRK?63Zb|pgp8THzxr9i0jc6n z3&~#0Y0Ot_%ZVJ;!sJ+uOZlQ03PP*!2zq2DJUU*WN8!3U)-yi_lhcX{vslGVU4ayE z!6)lUH+xPq)y{t|_QQy$3p{DGxYwT^mWJo!o6NPI$MAQo_F1@F#f6v7rSE9OZd4yF3+(4LiDyTKR5$#afO!prtSM$?fkpW*=!KUJm%Gk(gvHYmpSnK zL?3)#cb=zHHf_<*V6)B^oHQ_S)aX)4laP_i#sBJ7;sJPj=+vG7Z`^tcN*X!BYdFUl4;4+Q-rwVLV2EV zl-Hf*T{4%Bt$C`GEZ}Uo&n0`Rqdd2Zs9&L~X> zSn+i3v2TNd#_p;4(v8H;s7+^E2jyOIp5@+F8W$nO#${CoyqWF}evyQy*uXdp{l6|a9#w&{h#89M5-9in>$RMQP~Eq0AlMC(Y9_mQW0AWOMf zh3Ap*Jnks-cNoe1K#OZ5On7$)P0$2JNk~$eODV0CxfUh3YWgV&wwhWnKHy7}x$N=T z{wvWkR-stC%XH?G{Z$?U@7F{OEpq;X{kU;VXW>8L%0KLgh#H_eoxJ1poW$Ctt8XQ? zT;|*pvUO{-Y3Na|jjqk@&wj$p*%(UL6;h2#P4meWwt~jD0_t9oe%ej3l0!%Pa}Ei) zrWX{bnZn_h9x_E{w9uaE9g3Th-wg|u_3+sBP(FS7v~av-r7=yv;fM}R`}i?>`sm^Z zn+(>bS3Xg$)6;xMZ$5v1Dk7MB&(S3C#!~Ijb99-rvvaeveg4=;3jg;v-3F7i^ZlL! zKUCQ1Z3Ch@%Qeaz=bkj@e>bJoa*DTgC%IODoWlzX@0iL-49wFdJ97;e?P}~$KVp|a zC(LXtmoNl*ko(4a6>BpewS-i?ZF~c8|G!DSMd^xphJxF;6Fd;VMHF-qvBN*A1Y0HB z`;}c6DOW@fy#~epXrq8aLgp}htcSu$vONc+Fq`GV1sz-FC?e7JComc}^nKhJ9>?7# zTOMYGA!p^rFTfmLqhTtrxN#WZASUEKye8m^vS`_!2Dpv zH=@KIQKDPLAS@-aTu^vW_|9>Q$!T|F60F4pdPUb!NIi}8P)wRZ_ku~}rV4c%G4ZDN{5d-e45 zM5%Nu(H6k!qMp6!E#e`EgV_deLPAcqn&(k$<}jmDl#@-3)#7h#8r!Z%G+DAdB|C!d zwWf(X^U5&ZoMw=w+wvXAHcWel?>+xok>NNQ%L?+Jac}ti+hoQf=|{@S26M0$?XP)2 zJ;Z-)u3EU<3_Y|Hd}I0fK=HU{@oXL&wA^e$TGyRs-U#7YX=d(&ebgLv>2PiD(ET>I zlTPJ!IkWwXMb~ZzY55Dz4J>akyRS9+N@yQR<|7_q8J=${@;{X!Gm0hUrA> zG-1;c$5HVHul(`I9U_sR%LnpRDyvCHJsR&*QL4)j(vz6|gCdn%aT*CidL2yFyp(bA zbGV-mR83=5W9uo>0&oU$uvsBRa*7ZyqZeG*Z z&0z>9R0w^Oq{1^kETXb**mIx%xBczgBy%Szz>)ZCUpKeetiDmiGIq$H-_u-f`##7d z4(Aj7?8s*eMSrC{Hndi=0z$) z;`zbja!%^+qSeA*9+W+J(Gn+{#6n@Uu5I022DxR+^2n|H8TYgP8T)%oa+7c{f4F8# z)m>xz$wmm1Yox#2eyPGoNsEoEU9i|V*Lr49{ITTx$cS)8)ltjkQ-K!ypDo}BPO1fz z85@oNe+BN$!_Ps8jWM=1>w)OIdG2gnb=BL=-p&8`t|`nmdS?Y-G7tnMX4hye=spi+ zE2>CJJp6FJ%oRDM=B~4Bt-G(Cou z-r$U|j(>ylTlHTe5<0p{BETfhPV=#6EkIe{4OYbMxl2UR8rgSVWuo52)YLR;Sxs}t zV14jn(2Wac@U(SsuHs#~_*p&55X)GUD&y8@OZAtXK7wS$NI}r!3SFi7Za1p6E-WbK zg2E;S=GI;pgo8`m`>}WbRe)3-q+l7`N_KQt_R^TrCCGues{&30fo~OJZfD9+rL>i; z8AI2xyuT~|hlmlU2}fsJi#yoJEBmK&1+nyorp|LrumxYIb! zy)_KcENQAL_cY3N&-bGSj|$Sg-O|4jL6uj?EsLA;uMt~*U9Wk00cRf^8p?`ATdtW> zIAnmab?RGS40U%78uX&9)@|}A%mq3d*03yn*wQ40E_pyV1@LtUHI0wkAO6DBx1 zn#aK%h7)5|sL{AtkSMv`T?%8(XjL>S(ehCY7FXXL4az=u7;%N#YLfD?nf!Lhz5jj) z=!jPv-)kFd4%cKeFf0{HT$RBI!M65ns{N&>U;I2aQeYfP#Lj%bZ^c42u~$p*PYc(* zbVW)Pp6TLpCs=hSwAnEl!!8{YF}e;X`3~g%z;{SJ0J1B{v1@OA zKU7!?nK|9AN?20em&yn?%8S^BAE$Se`^+Qoy*t->qcq2QgsBup<+@4 zsF%8dK@xSV0m6KFvLyl-B?liLrKxduBOhPLwU+Ho4_8L9ap&o4!aTz#1_Ua%l4u*7 zDJ<(i-KG)aNh?2xu~!VxQdA67mCuRGpYBLTujX4DN@Pi8tmA2V2U??^oKD!Ai4JR1 zFSDLiah`rJiKZh=k-rR@K(sQe(yHO?5Q@A*qN-Nd;IV{pu1VhDm>}OYp+fmak8?9~;>6m< zLu7d#8Tq8J{iypzidnvUs+lR7dXVM_``b%DIYp8?6}t-CNtowEQRZ(0%`*)*10wen z$*eezzuw=vPV$1S2=A*kh+;|O(l#Hc@ydRC3D50%wXx1TV^{*4MjC3K8pWlYtZ47u zQDUB9J3V8`#3VGrU%0>|Hs;&*gFkiQyz_Wca@2CaR_+E5N%NO&?_TaK9x*X>?gy=X zswbFuZ#8>@+)pyvcMXr*Zl=f-OQ>GCk*KRZO6xye+|+ZOxLSPO)V}g>e>ZyLM^6-1FVo4_I!HM@=GXFKjVjjk)nHEnK9rI*f zLA>PFCk@@3GmFTGC?;*Fh7zXsaUUzOTjMaW4@z+dC@mwrgH$u}i)twoBw zBeIYlT=M7;*WP6kVc%-piUQ>Sim00wP?#0lLWyEsyC=eO00?fL8~{cywxDj@w>pP) z|4^Ks{_W^)8WL&m8U>PiS!?TT;MNKUYPHGwT)@5N>OR<0)YRjjU&e&i+^XGzzxd2( zgi9rTw9~$#;=}mECj+ZqK92Z&S9iIMmq60-GL;1dt#bM89Q35VxfN8JZXuX1n6;LL ze&*_yZKE)}l-}o8ptMVt3RhHgQg@6Yn)tanU~#V&AsX{|TkqK}Pd0Lj&nxf%)+Q(D zw*j}763_3t-FH%7i(xmEIbc730v_^Ph_g)H{ZG~p$5ToWcZg>ZzweNcJK0kh0 z8n~`L&QvR>Ty;XO*ycR}ADy4U$eWm?jE#-0=Y0gF-s}h0A^K?<8S_@@0MCj_TMnG; zGknR(!C~!<)oAj@@|JCIK(nW1V^@_X4T&JnG&ZI<+7m62c+nNEv{koVjua;i=Kr5X zO{h4exnd6M$a8TmB--)}@a*1vj@c{uYgv80rNVi=7f}y{sndym!uV4pF4JtYWn2)u z+*qN}d>2#qj3>$YPjC!8Lt2FbrxJS(?>p*v_Tl-KS^#zj3-#{&SD(Q`h@t_p%;B_7 zh|#k0-LV=zhtNz9x2GkfGSHWw7Sk&r#gbL=6lr5~>}8ko^V_J0S0iX8LK@i_+g_Ze zi(ocUj*x3vzr8l{WxZ~{faJ#yB(*8sE#(0HR1fyGAyz`uuq9?dw3HOx6jiJ?ul=E4Q~pJPGYEuZ8p`J9bswK}4DSu5B?jsO z#9i706PKGB%+lR!hmkjt9unzz=$va>_)ewxuQP8&Th~qjy!E_M2>uVVUgz9jp$$$2 z&jY`hkP&?*WWY0uB6Jo?!vzbTHfA!J}9_c0+hWMpzQmv!pC>9EkQ31 zp+B^&HOSHKsSJ8rxTM+Sk&;H5y$$U^G3kHwB0JfRuyI&`iB_*ocPJfJac`VT~e``^VAx-4EW@{in~i zO?obU@hQwnKsl5%`uX|s!!g`KGDml+EF#gGyH=CvB!{{06L4N$>Gn^&Z)05M?#RQ_ zGsDnK=NHV@3g`suikFqZtYSO>X`CV=A}4EgD$RR)dmlY~*j6M2vEbt4i-7J?IlF9R z1Ym6)&8K$S>vL?YaLp@Mv#qsf;ihL(7ch3+F7Id&R0KVtui_E;Kc09$tE!Mk2+slR zV-E9o7I|jBg~ll5Ypv)WFWRs8QKbU1OyIoV$P~DHHeqsdq25(Vi(3o_ z8isSFGM$$vM% z3Ig+H!>-NYsePqfESx$^geMEk5L5*Xi?B&6QgrDa)Ut5C3`rCz)^}vE$P`iFEgZQv zILa1M8uE7gtT}t=&ZfH8qM+wXe#QeaUnTWait;#oAOol`ACwO8*Cir>PbKqefNYk(_uQZs@_}om{jw# zj(%YN)_j`mSm%NoJ1({y2yZnDhm_^y(pfcLX1sRUk|;szw~wa4>vW!9h~&ik-a}^ z5oIOOn9lIuh?F>Wkhb7x9M~xA5u+JHc4Mpg24V0$XLF`*SgoS7*5+|V2tG+>Z1X%0 zKJEkj3S8dSOC4|+ofwShg8F4lIyCo-%@EBXMVn>yZ!Lg;`=*`IoIBa^h^rB$C&RZE zYd>lt%vYUJ_IP4;H+X-2M1!m4b{A8XW8+x?0f9HMv3xY`lco-H;h~|z@jjNwl!!>C zJ~sQ}fu@Hu;pVBD^A{Lg_oMy%Ja<=Xp}xC+%{;R(wNhM4KX{0da% zv}$8Tb_C(9ipJVJHz9E{$OlBrb;WZv*aldj*s0`vw4RPGtb zqdo=g-NorM`J#-6v`q#w^RfQcQ@_$^VaRFR_W{5X;dcwI1fS0R@eLt-!Aa_@B|J+l z*TfDaO9&@->Ie`8NZ>%8*6ME|oN|9!67dSxJ?Wlhyi^iRH)wTa&h={wKO0))#w(0C zFyv9;9I1?R)N@0pt;srym-^`|DYtxQo`s^kNi3-3ALE?O*>?yXdc>WeN& znmROy3GG>WrJW|jgjfSqZ7cA>x4#%ygLX%zFeAo-FF28+gi)X$NF;Fdre)ubgYB?k z%*{geCW(!;$U8XHwqI~*2IWOrynF|C-ujux1Nyc|LtAnu_Me6QhZ_=uV%jbTvSY)@ ziXVMHXkkXqVqU9hT{+r`&>xNDOw9X)1l396>LSP&?1k@-bB_Mu2bf55^7ux~B@VUk zYeWt@#fU&~f=BzL;vk5;6w4iprXjwPO7FyU3dM5R8zuC8W|-sAJPMP0=q!}`7VK~< z0Gd4DcEy4G>p_#q)KnYdNp7!Z7$6Q<>6I@pSTN1~P>b)0&$zVSdd6#gyf)LZpufPQ z9`HZS&K^KkwWPA4&5F;8i7zPK!RYIngKO;oZ-?1*0xb_c6&68_13d}x1}1hlxXBt#nC#NVMDkBa!6oh@nLPleCbXi%ct0tbDx9-136|cuGIPpwz<=KA{BR3`MGdH{n95=TcNlH`a(aE5TF}16&H~F3(ZGKs&4wwB z(Xp@ouuiOS;N#mEtQM(X#ut0$Q%>*>&*N=nFd&6gEaZ8k67 zGTVu}aZjX2kXZ95mgW$T6EIfY@~ii|HzS4n?nnQ*-J|MGL9bUlD&xH4grjw>aTVam*TK=qc!()1h zmgTqQ#Fy00wMxp>m{1K(J-nzt`|jjYvG$JCv*Yu8UADJ{d{ms4=E;A!ppKle&zoLV@@n?n3EF8gk&`6`sApc8#Pc3k zzi&BdCrtyz#eY8ZuPC%dT!kOmr(tws<- zvC;^>RxhzxsJ>N9Qu0w7Tszc*0S`1t9lu+ka2+Kt%1#8>~~=9vQMABIWDglcJkt*@vKTY zoA#scH&LEyF>ene=C;QsC($wNu4Qp?6-RAy;Vpd-E{8^?Ny~_uPr+j1Cv)T0Szd1mrb zwbtC4DiWKIaS=H(u#XS}Xb|7SJzWzF%4eznG+u2|w;j4`+LOtW2khCeebBx1_tIt+ z(W{xM5_3%M;X4%?=bg;gFi)B%6$zY6XQF@cOwF=1mD-wVh*?Gy?wj3o=R#vmD%xPW zJxP~g7;_GrvFX0_@raK&0F4_hAC9p1G=uA#xECbu=m>UiVw%BkpUMuF7rhAgfaaLb!ow+*H(PKdnwdakl%jxvrogJe(gtyby+Xz0go)T207k( zssEFJLZfrYWj^(k)vXkatJZ2V-Ts@~*n*SkVrb1H4a@&y2Y&hqCyobcJ0G+wqyDGB z7*v}$VirBL_h6qgS1@-TR%fw`QGS zwZt@-aDXVmm{KGe=ztKBw#Yyg>t=2zCf*G%TP6)9I&BP+>-{PZD(MLh+j&T_zLos; z3lI)MY}k6q4qBei5Y{PT&$uU_yITCKc`;=)yQnYzSAS!nykn22YC00^muKnzG&D*Q zPJGXC%D~A7|NRVq)T-zXjRDRv=SjhhnJ}UNVMJ!sm7Nf!p{THhzMh}08FjW7En-Is z;t5TWGA$aHXfh(+IH%A{7Da5ESH?yiLIXe~YI(5`xsmt~(4vzh`4n<9C+pQIHU~k| zNAxfX6n2c}31tNij*wTc_&jAw3sG{P2FmeqRHbozg2&i2_RSaTknGlSl5{efXY7+2e9j~$eAK)5{+Xv{{nuB>bslv-zS;b5p@B6mzH0V*%OXdz=_toc)5j) z%dk1IH5btvwXT9O<%s65#~hYN*tk0qGyi!~m|IgS=-nOm=*Ea{{UdSR_gB9%T7)8~ZI2R5 zSEv(;7MuOCW#-)zXlnC-L7#SgyZM{bDbf~s^R&iv!x&KbG-yFNCV8x#AoS_$>kGek zy-ree`g3aP8$#~}zzl)Wph2Jd(7O*GinNC!I2{`2f-tf>jE>hrilROo7X3>qfD3@R z*Gi9rN~D@*0MJuT(iu0~^b4 zi(c^g^TSk7;;tfwVy0rY98|T%Kc4y@=Y~8ToJ9oQV1=A^%6+V>>b#IdsYATN=^ZBkzmm)r~`&O7^=Otg)Ict(2>-%nhktMu?QNwFe@m)-owd z_w}6cJsI%^vP6MCv@Fh=$8m2jnzEzqmx%lG>AH4JTGJ%dY+0J*UG(UqkB?p`{PAT@ zoLQY6Cs-UxO5^Hgsnn?(4?0@=#=y)?s4Bkf_tpJ*;Hn#VkGNb#A*|Y!HoZrL?nuNDKVII^q za8s+4n*|21PPt>ps)T=O#ttD`&&YNqK{s*@rgRWx)0e1l4&6H5#bgBW@~>pm>mt>B z{KK-S8qPVhGtqyfMGGMQfJsXuzQwpNLY{ZIm8lXi z-qK*L>x1JTrMh1F<(T8Ypst}clQO2c7eA`J` zEmg-!Bai%D?FSDjWkvePUNp_v7}}O4w;BLo#X9qYc=t$4{zOt(+&%WJkAEJJzq6Cy zKe4cb6sIow6wT=jV*eG%rExx#r;hYZLaU~UY8i_-o`>f;i34A;iptL9Iist~6HRZy zwNnnXDr#z%Z703~SAV>|q2Y~fnSC_kxPwGb)ac?AyTNn6fd^1n9lZ@?-=MXK0Q z=#gNbLuqT#QC8l*8I7^{$71~b-v5uEq=X>rA8vO(S@$7@hB+=XzGnmsGb?`9wkNel zGuFMDZC3oULXGC8Th@eOj3No>E8BKn_$jm^R$0qcIM7nUNs)z2=;W)o-UqS0F2rKcS7zWs9N8A4*=tR(E2H1e&e2m)3P-Xu$8Br zq_8T6f`0HzLm#d@F7uT3X2s97MPr{@NRq4RsDsX$&dyHZ1gMp|vvZ-oz8Y`O?wk_e zLa2y=n0{<=wv2}w)2-_evwbORf4JT=kF24?YAjw}SmJS)F%qFbkb6_Go;b*_1*jqP z?2H^uHK(dhl(&|xne04*r%{gnYJxe?(HwC0{(2wtSAv7grv73B=zz(1qkE&1d_@j# zY$mwR3D3VF?@@EIh+km2y0pUJU!OfHBL_x_YjUs5lHPl zfZo+hTc_y&(fg?R1BXA8^N0IehLRhY?h|2*`UQUD)z&&zk>(%s@*ab!HLF-9y$%Tu zm-M?aTP^ch{YC|ReG+7<5-u;UPh-PXke13-?%L^02#eE~y9)_-{!mwR{%cRcDQDra zO3Y}!iTz~kyj8q%WKK-AQ7b@SS#}}3`v3M02Yn&A)1~1#iph-nRr{-Ol!W$tSHqOA z)LF<%0B)_Zl)_#zhX+Ux8>XWC+KsGWXBRHJMeOcCPTdx~F-?*Kjr=*!$8p=hq~_H* zaGgMZk*=6$DOr+SRiJ(Q19S%d2cd4Dsu;zqCv!ZMNlEU_N{IEy%hP^bnJFnD-@bjr zNJ(D2%*ole(2vy@ z@jR}0O8&SKqDSi}-}Sj(>u{vRzCIy)rIYyzY?0&$+t&s?KWm%LcOg%8H8BL0Akf^- z@qPrPt-x4Q^piek1_zgH*4HsFZk_Im?TuxiuDO$%mKNQW zu34-CioWWqDm&1G{`Cq$Xn9h0q$*u$N4L3?CMHzT0+RJuvhIb}Nryl$ZITB2(|~v} zoU*R=cpoBt;Ny^MB3O#9L)44!Heyw0vT1K@MS^2f?T*=g{_Xv>+V=uiuRa6?H9&$* z$33Ax+b_6Zkb8C=bj|>GX|OgCMkvLBX3_PyvghXq!)+$+Gf2v1h?U2>iKSUmIo3ZO zBtK0Lxy%H@Pn?5Pwl8nZFQ}H~8bPg68iS8_yXW z2if|h@*;e|cg^`T@AUxOOp2|$Chj=05cWhmDl+B34fqX&ep?{3FToeZzpZ?9;Sux2 zPcoU^YjXpm+im__RNNOF&p(Y6efmJD(BzBjo9O6)qMVN(FMzy6!Z-;esm<1)4uWiX zjfpGuIA7qXlY0eio<>vmsiE*ZXMP!ex@Z0NIhXGW{I_on-KChjgw}e)T+48d=_zMO zF1?r$=!HGw3P-qM0K@~Tf&_u6mcdw2`_yGM*CyhvS(ncl84fWhuj-Ju6Ze){iE^yf zmB)1arWFNw%T>WZrCtX-pbC5>5mk1c+;sn^!y4e!sXQ#%3cA!LtL|xal%b7zmjAt^ z+>4r;{g!>(vn$%}drnAW4}tp-GuPYj$BoaiA1#$In^!`WSmFO84f@N)A3{`+ub&@O zNoved?ZkbCnm%&Y32**P?)zFQCc`cUp1(iStc4=7+8=k*vh?!Zj8bDaIJ3HoOA+;o zy-scgir|(GU2!d2IGJ4cuAjZQ-=#^WMTEsdOsJ&IC7sj@^nNJ=nbCOvsH?oLqW5vd zSKy-I6H^)$%^jfa3)~I=?WhTyF0VTb+!sz5yWnv4yI~8O!ip}T(mikf%)Xq}I`F!Q z+Ee*lUvc@7Eq>i>*ND1Va^GApcS|4$5{{L|1pZ;2gqOpwrI_aI5-^nco37WS4X8V(31YRz;z9wFXP{gz(cVjau6kB&=KMxd0e7ej0jdYet22>EXS##We zpTYlJ1T^XQl{#FWMTh9OIW01#f=5h6nrWSXu{OVV!`G)mvq%<#w8+e-u*}^G@;B~% z?NF7!X#!eU9^yP<_>nBAP`7a=W-Cv{HS@cRj+6SMm(L5|wP0u>C|BkjgeFI?9Q5b^ z^*i#zgnGv?YxY;!lL78`wC$GcunL4}WulJpAm9jn*dx7**}fU$Hd}^4Ozy`TGKK5E z#-dTrW6C`}QwSBC?q@MhxvDQy2~~vt`7rY|TGT3V9ZU+18Wyq7mlgH{CUP{*0|TVpE;qvO}z zBRkRlRfdxuyW1PlM+(5DLWI6r*n9WF6CbBFNS|gBMAuB>u1~P>5i?{^#JiVQPX;^! zepm=JSh+0q3jRAgpt?g$b2&KE00Eybf7((h#Wa2E4$QNykM53#**^+fAC-sA23DUU z1Lt`pV6kE>va7@9nIm^G}GAH{8itplMggq4cJ}tW5s#0n(wFA*kyIIxo-;~ zlk+Cv(9d-Hf~UWo$H6~+Q3FbWD-zD!;`je6Mxd@03f3QG!GSsC7mjL!s|&6-&$&gVk^5(5G6~GDP!6Xn(k9 zMDHE|=t}{hK>(8p2;)|c+K6Qp-W=2j3W{DT0e!9$bC0Gb{eI@^9^3SUFo;?s--&B@ zIWh&x({qqeTovHT><#k2uRJ6bAh6$TOzFjMlSUoj6XukPq^ zx)fa(x$~)$nJPfUPLQvF-j=(#U_uu#>)fgLT2=}DHc6#aCi+6Q$ngc1L6~u5 zm>3c`{5(cOBG(`1nS&`m4kd+B9(EC_KRZyk$BB+b< zyRObUBsT-j-FrJ7O4UEr?GJ)sK_$D zl-a$QK^b#HX2!Ai<)aq>zY6Q>0@Vvy2%%`};Za7&fPr;We-FPhds>8lvfgVE8i4)u zq7byc{O}lMcN;g}Q1pm7t+B-;r=UPl*N-^cd9xcpTXla6x~~(*jBWk$3dQ!~citwo zEZQc<9?7q+VVBh?AKSUv*UwO!9(lbLsd#wVRMI75KuU-XR|01W(;gJP6UcB9J=BV_ zfLP^Lh`@b1GQ-C@cqCuZ7$Xi>(b`KC*L|6!zSt`DD#&XOfa1$vu-lXMvIjl-0H0M1 z?}#EYb2BTHp>Iu#$JH++`u7mezq|nRS>nU?Z$}p9-jriVu|`^ z*UA{z6nMaeBXj38#}W_mb$T2ry^mueim7F)+S)2wuKjSq{&?{`_XBI+@Y2+|pgfqi zu;?S9rn3JTatUz&VMWkJ^*5izy*`aNHBk2$d;jlnd#7R0yDW&1f+~wb9W0^K-^}C9 zXzId4LabY1+);4Q7Wx#l797(DDx-G7)TRw&wU(csa}E^p4ht+7g@iFHP^=0`X`>Qq zk{Wv_HJPWU&w&ck8+s8_-&c|i^Tj_bbKQ%*>oqPG6L)Jcp(*roYfrchYID7KEg$?f zWU?-J1y1a+6Itk2@ZR-S^gvZ-|LpSo()YcFZFL!x>uXj@rLbuGCTTAc?oHe)M~UXG zO_37isyUl)f_?!UK3>g%znPhwlIG4&F&RuQYEN;q1W zz`r*~jhKDe92P_#fDo}|blP{iJ!PbngHPnn> zQ%|P9;*e9au@q}C)~ZFulaQW%AN1Tr*YZs90Kw7v_VMBRzEs4sE*3#OM8O*Qz>-_$=pU4BjK2};A(vb8~%XX2?-KAsC-WN)yLuH*WKfr&t>qWUp z!wE{#=cIZ0-Go$0ObqL6I@6h=7=Vqdde*pDZ|NkQ=1DmOB$d<>Xn8o$P2qRew#B3) z+qiEa*M_fW^X>j9eaf0)^y>IIQH?kL$@hi^uePrRGxA&9Po<9(Y(E;1YltvYnafW) z-)u`>u&`5_hE}J3q)e9+P z=;iYF7h5FE&pmZPkDg3!I7^|HxS?I~<@j$}$#6m6_{oN+;9Vf80>KBgWXu25!~d}? zV3=MF?xSbCJ%aKjTC!&Mtx5={_TR^?hgi?0vE-3ikr8bg!>r92Q%)4CkV-+$wKB#Md{yXMzdDT(-+=^@Ldg4Pn@$}tSXWKT{ z2^ms6`yl;P(w%|z-@4FwSAw9-p33YjV)$q%?lF+%wM4!@D6F{AW=ZarE5l!eqsxZ| zlig!GH?uKhWqi+1{^Q+={j-f`Mpwaa#WM86rxPUVM3Jnop_Q;7u%d2{Hl9ABo7a`@at@F~{&+(_99) z$WAQUPf{PZ>~I{QcTt+lnB-ZulgY1}HK%;F+C*u8D;n_35PdE{HD$dOj{pUJP{?j4 zWJI4o2d6LdMld9JFgSFA88M-sp22$ON(NZ$=>}UD1$LYoU8RR!`sG&Fm|X53zP{nP zI!{uPIy1Ak-YiATLQQ`8!l_iXBWJ8QpKv`ef~TpSL?XWfO%Tcpg@J{|sidh?stlQS zc~@bN^?AcmVCmiQrftVQlPey-@R8pST=S>DzNd2kX@yv>`zMu&pm2=}9GHEi=0`z4sm-e+E0 zt1O#br!Af9D*vfCh$|&osZ&^c_4K|k=~v~FrWXd~$pLnJ*#qwkm1JK(j!U0AwY*ZL z`GUz0iaulUa)R-hexH$P&Y2fPC)tpg4WHB5gLy{PgP*v}Vsy-3p6H{Np?<>34Qsru z<=B?N662vMP-k&$NA8Bt#aGaTO|dY@|nhR8%f88~CC=P~?Xem}|_4zVni%5M0oa zDLmU&{uQ~?WBNkGh{i|B6Q*@1df}sqKLd~J`n%{%xff-qDqKaV2|HD5U_#-KUQX5U zPthmneiz37bL(VD5fZSk0xux5NM}lji=3WkhEvzA6VLuBbeiL_DyGpPEx(ZO^a35m zX@c`Hs=9Ag(7YqoPV60pb`52Ew9WjjC&6yMSJ;N#wC?OB58fAJswpj*2~)J`jeApZ zXHbx?jPB=Yqu=Jbv%p??Agw2FxZ4jiy%>4^LCQJpH2YoW*SB+zb9JI+PfkC({Q`X_ zL5}G^|EC9T7;`=GX-j#2;SLt6n-!^nq^A{xUfbaWzEPZCBPadntpgoFtxdjm#0#u? z$qi(GykMph%JXiqe+@xG0;wSH{Ze& zKwbsXeg_McvsSZeX&kF}a3bs1Rc%7Ww?ilH3@=x&6$EK>nn!-Wx?`0S5%mfTQU6Rr zJagv<>htF*-@j6;->`+8IanVN^I;+4#mh4TH@@wBsOj^civ1Dn6aNpSD#c0OQ+nMt zK|}UNHFogfJj=-?Y&kOUwGBmOk$qcbIaiq*3>R3MUVsg{F}CJ9wZ}QK_cGY-Z8PPA z$1d;sUxZ_Q@ZJKhrfK7rE5SpJt+%xuwJ_V{>{zHh z{OXfXck2i0axfu-|Hpsi-!EC!y}p~*(-=vR+=RSpP4W6TebnW2z$WM8=wql-Nyn13 z!K^%d;u&ntNA>s*XB_B+TNxEU7r#Gp+?aDv&}GHAd-9#ge3iYuMvw__mT9Fv+GOJ* z@)!`wm~*b8ZMR0CM#X!siwI2B;lQYQhjTWt54h@b_bpAW%?cs)msdF7KGDn59((e` z$bglY{5FSFdZ^lgPG0tfRJ>u~{i}B#4(s0CVv<$m`p9E8tjqa_X&_se9c-u-|Z{euhB98{$Ir^)|6(%w2O%DrnFRs;zpR0b3z z#Gs@UL>dJYL?j)$yFr>^fI$fd2|)pAk?tN~C`pyh0frKg8ajp;2EL2Ex6f_8@4mnH z`ETGD4zBALYn|&{=UOY`rsR70wdc1P2+k3@x*J^~N^$M?RP;x4plU}w*AgK|Go2Ev z@r9|=Eg?=o{@t2u6s)E;v*Nugu`u}?JK47fuzI}PySzB%k zXw>Y&4zJr;rwv?m;^strB2Qc^ZR1*+&}56K3ZtA$nk&zB&t8sqEit!Lq5Wt=l3E}s zsh#1jy#YynvRdRpL2UhfP`beL} z9MFT(2AHk3J>J2)1X(lwdc<&X?HhUCYtL2pLusI`%2_G4ahg*!-xX9pI)$HY5zK@X z@C};1?aw8*2mpFbMSkbwg+xb>nm`$esz|KP=9C)3`*Temk{&wxj(aUEuwms66-h;r zaqFG!A|oBJuQdSpSMb%rV;ch4WNA8rPQZ3ssMmZ=0K?@XZ&Y5oy@YaTHIq4!chNw; zZPhIkZ9aOHs)bGIJ;AHh94h0ua;=HzrkH>gpq#BvQc{Y2oTiTH>aY5?;{#NEW+S3)#z!14b#_5G!SSUIC1}qF3;W)NeF`BOC zTCl8ckHszC3|j~nV|RC=K2ykgO243R%Hy5kK6=||)xK$D-@+HPTvE_;2e9^>G7>+p z@bAY;lKJJawP(4bdWGt_q$D(sEXU*$3-HrpVorzC-1{|RlCJ~mX@O#n4#%9Ngx)PASbwwml(Krh z&H|wDuD$O(O3GT9akTD^S+5P$S=?uQcIgctNj3C{O9;t`1e(UCd+k7_*O?`6v5{6>TUpFfp}`_D~ezKm_-MwUxP$ z=EW={(xmflDbK{uU0XF5E=RHqHt6?o^JREUU|r_03=D-+hA-_tSFM~cqU3bm?cjZh zS{Ox%>1bp7zd{;k5UfcZ;6=b`YMH2% zxye4A&&pWMH(OS#9So}+s1dw?eyP~Q#}1rFINIoUm-fSWI12?FCMC`FwMY81RL5R8 z6*x;&U)Q8ed0c?X`mpNS5v)44e|@J)_Ys8)y*kvf2I$E$45CGg!9>c@c5y4*$fe%F z=U^Sj?uv&xu6BI|99%DaK}I_J^%OF+%Tlh`xlxX8pq*6~!O4yX>*c8tw=ex<;{Sp^ z;e!Mj7CR`8YEh!JN7Ft9yCE!5l+qajqH#YmKrjk+hY~P;^R7Ch?fM%hwN8tGSBL ziT!?Re!0r9)HA>pSeSVnB>9-&!T`jqE$HOKCvM2ILR{&D43(PzNUw4^WACIX~A~MMM;$oFT{N0{0pQC6>*305v6=w(uR;nJ-=}dOmGQ4|H#wwefR;Uq0 zRtViJygDj>2&3Rf{+Z~ zRZ`X4fQ){LjjB70+I~9w5B7&jwzS?ccG9yM#&{!pU`T=^U)5{n3Z>LfR~{D1vZlzj zl}Qz>n>HN>x+)bPR_-464dLNxHz3@tT_o^PgNf}(@80ZP?N&zFwbA#zwDV2oaJzfL zUfQjR@@rKqPY*Z!+6{L?6XPZ;xuCc5kcV|b+0ObP(vKK&8I{ro}^>;M8XvARu5uuj_E%Zud8$^wwet2T_- z&DzHd7HaZ*T-whjq_}N4rUvJ%1FO2k58Q@yHC~qkZg-{I956xob1S$iN0dhmamJ~W z#r-iv<8hGe9332-{vf1pY~x6D37E$-_T?>STz@-TZ< zQGozB=kym$5;>ih(yz5M*}z{45ABDK-C-ES;A+WmZdNR4wd*|Ow?3*0yLz#(+I1$p z!f-32=Tn+EL3qE%Yw4lnJBCvpUmR6}M2l#4N_G^yvkK|S^E4*4$v6+q6gkT=c`Qw$ z)b^WS{jdG0RO7qj`kND*M4V_wAJAN9B6xXDBezsR@b!OObJ=ag(=#`a3T;e|V#Hzg z96@FK!|W@_jDt^Nhp075)aAPl$t4Y1T($rt$}cVL*TvVH2_6YR?y`}?CXYR|WmV^f zOU<$fGrDGfE_=P7&$V-@QRrQi)E}vbE zzdW_lv8NWG#RHz2_9SlXgL@p$n`?hU+ZI??qpCxeh zDZq)y(W)^KCL z$4+6Px-q*8W7j(Pu6`0-UiWR6)^U4z<8n6XlEA2O#!K(?B>Qc{XAlM~U*r(T3GY{$|U`A@dgSJG!cjL7_ zYN|yU09?vuv2z*)>{VsdeVg}D)-oeNC_L!txx4n&ZhflRZY*7RXRNnNce)c%oHkjj zse&WbcwLlMwX$g4Hs;1UKA#w~Ew8SkG6l{rC3ZS(e~Nf{6$~|?tUE~AX7sMb9Ody8 ze8i#^7Tc?5o3rfj5fC)@r^NHZd@kTmkEmotV7{)8*UL*G!^6D4UzBj(lQ`e45csM` z{8zbrHOZKBZ1(w)T+?ts0_6M?3IX4=^XY7~%1zs9* zS?F1yQ!owJv#SqYIS96MJNZDfvKibO`x#^m2K+vA(Ch|bL^ixorhO7)4|CL!l}>Y2 zk%|=9LUok0f;SF5F>VNX-Xb7b1bUt$~!# zIbzCI=jxxremx>A*bhT}fm-h{X;Yp~LT0@RMa0MWlUI~I1=Ae)@O>XjP9KVXnoJWe zw~si;l|{e1H)~Zud_OU{Oyk~qLYx@KAlXQsqF>uy>~tdYX6|571Yox;$qG=qsRvnJ z<7B%X6h)HA2TQPakhpEV9!5zqHqAXw@l)!bF6+;Z;edMW+0~l7tH~tl2ZlCIOcJQa z+=RWF2E}(Kw1E7}>%K9g`bQ(*wE)~wOXK04s1X|boZTIw6}-%*6ip8YK-(F&hPr_3 z^U{t7ti6qjs_Fo3uTxax zDE^vU%qD-A+`ZW<^;T}#Ub>MdPDt-v%W#j0!6JM_X(GeQ8Qa@)#QWY#5rb5AbimS+ z*<4wrPCphCXgDBv%RQGa2&s7L>%9Aj!V)F z#l$7My0Mk_`zaxeVk1NZu=G{)c1|}+aqV}hPj4%jQ)yt=dh$IRMeK$gQB`63l>@fJ z#m#v-mA!y;ns6);M7e}6XtK>2>GiB#y%Jpo1Z#tr?oN$2&IPYJ0waV*T&u@4?y#)9 zI1A*e>Ij!4!oR|K5baXycj}rx|Di>UGX+e1q(y)X#|#Y()FwyOO(3>7_3995lI~ z5UL25uzh?rBiqd~{9K!}jm6`4=lh1+T%uz@*LiY47HE+s=dk&TBt?Dk$9sBYe5ep% zZ_|zPj>mtxoxl99=Qup0yS~pHu0_zS^+b|yH0teJ?U{gtZFt)tszfllal?B0Wy%0Zw4`sFywg&2u^H5 zwed^T(G#mmZcQH{m3y0vXEpB@nsOmVg~xOcOXS@&MhZ*xTq^zX7ivb|wCf)`VCZCA zzMek1w5Y?!HEkwVlbXQtnt@Lsz}V5%O#Upm`ujw!v309X`(#yU^mIt&Qwj`ExJPBa*?^s-iCHP7HW;J;Lg&^GhxaADG<08dv*kE+ z1}Fo4?UiR_`^XV8q`7?3tmVPRrl<(5UCE}?T&D|=e|e@lA#wPjVN9%!Wm~X~O!ZOn za+%wgH?5-+=}9yHgGd9cklXZ=;<%Zs(Qj3UoX51*5zv<5MuEP9JT2Z~Tdms0ar@|o znXO2JR+7|INy!`lS3j!dze)C!&Hl@B%Cf`m=gr<~rj~&PzJ=x%q1X>o3ylc=X`cN3 zg2Q%+#(TQ2+zYTDEO$?MsPB^01U2YL>v>2!>Z6k27G8Faal@@DXn5&C zW91dkknZ*C=Wg%zeHBO(Vw26rIWKwnq_4ecZgZKloN{Hnyy%v0syQ(z*tJ*D()Wf_ z?V}`a44)9@^!g9a`Lme^hQ<<~gd_HRWyuIfHX z=0fd}rpS*a(#la!D3(AlY8ny?TOdsTN*k2+hqJSMvs+;kP3-5BAGG!H$U$DWAX1Ry zP+hsN6Jq4~69bT3LgJMgvPD^Z3HgzGXuVuFsh6_#^d+QHYwhc6$&LJ7@>gbsHCpu* zdxf>qD^vC+@;|1R-FMQCEKI!IVc5L0;n!|@bmz$S{2-Mj84uuKN3%S~^}@>FO8(>$ z1C?Ub+v0YUI^T-@pdVvF>t|d#Td#-z7pu=q1FU{?ii|#?iej>weC$YZ?%yGBQQ)de zA{e;zd3wCfwwim+LkbDby>cSdPLO&1w|4uZ8=a>KaVcM<7=K9t zka#_z%v8fK!erEa($eCwiCj-a{WGr}eOne`hfc2y`zg z@QKz%d8VG{B6+4hMxTEn|4;YS?{?zYz9gR_GqLW#QibM#@M5_WWM%j#j*%(tzc2O+ znxTAE>}o{<8YArBKv>b`QtK8jj$IqFj(lN<5#{})!?QSa4lv(n%h~zB^?S3>C#|&! zsoNSSx7z&L%M@BY*~@QOGJH)3);q0!V_K_$o=)Ia`zI;qm9mOkfs9+6302}v3u^2$ zbDL3ezc#b@IchyVEzUN;p>|n9g$-OaAOIKLOulF*Ve*Z-%Fw)SvDhhCIN`~_ej3Ls zf#Ik2?inuj$0=PPyVB=v<%*0<4QvrqF-z~4_szq_X7=-1n!5*H_)*teBG`Jej@?uVk(g7pe*6#95Dle ze>*{rub?Xtp*@a}aE@R8`*+{{TfnN|PYcglp~3 z(`-t_PUll?#-~kF-Y_J8oxbKdCi5l9YHqo9|A=GheHxHb#g-8Gcq1m*zbf9pCZ^FU z1(Mi;FwFeB#)T?K1qeu$@Ar_yD3~3J2k##dj-QO`IDbx77uH2PF)i+$ z)&DJLZG^}?#Hr8zppqUKw5VNbp^Qn%5Z9PJtS!vmo&%B-&sjyMfohBCL7qy~w%CkH zvuPGmIgB$$16K_3S=lYu1Y?LfIW%Ya1R4R_#RulAqEg-AqkijWRl#3|kKtauxQ6`tN+6UMh9AWqk-r0( zMEHh=D9p?lKAZU9G)F9?pKq){*{fjCrnd48e(&CFW`7YtjE#(UU|dk+m@t>amvp?h z+moYRRthyD9}FLfZ;J>@Ph9Ch$-l{(|Gza6`Nv>~p9KuZb*XnkRI-WnXB7mpOsl>V z!X72JFAyX=PR=(A*7p1x{ynML;@cU~_gMrN=q&@&1y_zEpF~}qW$rqACro}DSo9kJ z$_p0ej#&XMcdP!}Jw#IIX1TW+GY0%p|KG&Tt05KTe-X>da5R;k*1ar~Q0&9B zGJ2SEinzHpKIXxJUJVZ^-xRU%VRTIK9%@fGt$1qYqt<&0@UGy1`YovbLq_t~M`6N; z5~T4WQT+g!qA_c=S}Rt1f-@7v+I~N`+kpAr8K{N_>x)k{8KJ8V&-MHUCI8?%ezgv; zi-2YLD-Ao(OmH%Qa?lC-O$dc0BiQ?KR?f-KfTyRUh6V-08>?>ecmSM)qib4lA! zbse20v(EShJ0H_GoCQ`ETF!vwUcq}z>(-Ff{=-aJ<~5^_(Z%v@?*1CrEX14!xPPk8 ztbKFe`|(Yod7$UL9BAn4DygAaV!BhBobIK_KY8q3E?yNu^Y8fQ}gOFzJ+7n17@v>&Kmwq|M~j5!R*mG*&RSbV5BtNpYr>Mb9VnhTLW7lb)EmfQty( zR?48O<|zU0H7n=K@-;TiEPX+eB`kh4U|xskDJne*j`JJLUo@*dxtt zylc>5d5bOIOv*paLVpEs|0qQNT6W3v#}1CrBwaF@f$)LF6XeT`ESE7;2YmPK#IiU= zU>F@$VEy;B(HL$C1FpvV;Wm%PR?{E9+j#r9CNODr-=}SnK}fyKI}747?yg=iWq)jY zH$2ge8Y1ctfMHUv#C|J}OBqs~^L#n1Fh@;dp-pX%i7((n{>a?GC|K421Y*kMwe}Io zf9DRiHv$6WShq)Op%56<*1}@mww1c(_N`lTdXm$YfOhM~7L*oRcGRkx zlL@x5--JHzwD0Z0446N!1#I*i#s?oPts6P0=uF2`5T>&eA94^i~G~Yx^pc4HG(#)C;NVc zB)=M>e*$L}+4{DlpmJM6>aTE#SGFsW_qEN~Nm4`VrI($}+Uy|SP%o*^J-2W5neEt6 zpG+bG2Edl)jIV{#LmDW@>Jq)W6I{kvb{aOjkRxybZqAHu5On!&wDwS)etIjE~=b*8y;Q@N4<}VE7sK2FA z&odTSu9c+$d`J(;Qr43+cnP&0#xK?{HME{x>rp0wHR{e0gnQI# z`9Tsy%6YdWH0hmh-l48g-368cJbIIuH0E~H8%S0OUi+>FPJ@`3jm}8!qRd26PFKg3 z{k-1wk8m|l#ZHV1EzwIJ-F=R>x9~I6m9-|Ld@n%~2XQQ~n@>-u9pdljj;-Lo)x;83 z7pDI5-oFw9z|K_z99Ty%qwzUeC`?mpYM86-vcrrsqAk2n#b+Rw3Pw$B!SU(Nj@mI9 z^C0ZfbomN2(t5tCm4Y)5&|?BnrVCm-lbAmJmpN+@1Hk|C$ApB8fdL!GHYV}%EC=1V z2OkGJH_w+uc5+d5V;eyFcEe(IeF->W1GgXTz+{OpS+c3kSk<;HR7^Q5Y!D@;wwoJV z>)M05DYClx=H+Acf*Ew2ame72V)Fzr8i4lNGnHpoeI<1Wp!|z;{!m^a*8H>jXNxIQ z(Zu-W_mVEZ<)I{*9SJ1V7Kx=&5%)*2t5pumgb7}^q=`9c1{I5XTEC&8!6gmt<}zQv zdw2Ckv#P9>Ok-eJJ$<$k_gP%kX)drcIo}X`SW_9PW(D|dlO@wPH;RatoP%C$;SGU6`6w7)qOtH z)*fm=kH{5tQl`uxi@54kt`sPc1Q%DgUK|*~(36vscTs~tVg<(3o>oby=bC-m$rm_pd(bN7vfcvI1fZve_h?#xFP{-GB9`15@TPAaAq7GG=v|kzm`?49( zFne~;Zs23m9)C27|IFLTNh-Oh`CS+pWC(^bTjAlLAbmZ*J?jL{laxP#UrjnI` zo#*;ORskBGx>d4Sz6$BY=oUN0c>~o*lCKEBnH^*m08z80{=f`~bpfA_**{yC+(#pq zt5w&}l>~i^u0e-bVCNodzIx;Q@B|Edj>KZbp*dO`45(YN9?+Pd3T9Q5B>lgTmD`Q& zD{w8#LXWthVoLf_&`$zHZs8TzvmDT_nnT3$N4D>i^tsvCc10a;M_huH(z3m&H-)pq z2`K5X-YfQiCg%uGsy^NPX@qpqo4^6{`$68|CJRJuZPOEIJrc=2-B`#dLd@pLuN->fl6lF zt$$9Z*(YV5J}VgigpkwSVs)%^j`3<<+^Bno z>jq34P)$3^FUu1go(NCGm@ArYKvo#o;JYw8VN0f{!^)HlVh$YiAYsakh;nA_#OI$y ze*i4W?~WG?h!GsYxrKJSbGm}EYf>*@*Zkb$#MUMY;1?RN-PN9B407`$d{tuPl*D!) zB!}>kSpA@}YPiBFfJGU$Xd5CnMQ0Relq3MnN3Om4^|C;^RT|Tc2ixe(Orf591|L9!w|N8^`p$sm%nCzgjY3-P)`3B&e!WWy`&hPdN27D$>Tw~3`A~8YjR}v zX+~TfwFBk^&t-XO=BI>rxZ|e564pDO<#!;xD?w_SOqk>5y?<6?FWgxB%$hU<)RaH5 zwy~*}?hzQnCMDzT)W-wA2~}ohTB!QXGIJ3mXx58wdB1KDpRr|n6p9$Im>eU+QLP8K zNAgnqOsOa95Y^;2=~z6_l>_iZsMd!m~c+J&mbrL+%}$AJ=x8*C>?Z)3d^;x0a1 zk*2Ivy>M{1{Tk)MXxNla(^ES0BSy-BsD-9`@a(&rte!FYsDug3f&cl#8~1xpCnSB? z4W81JX_LBfYrCc<-F&y2jnjMgRCDazjS>7o`M5y`ymK3Vl~wA$Sk3b{!{?`Lr%Mk# zNXJht04UA#Bn~Q?m4-{400wCkyf-I=sOWE9S^!4iSzb~4F^%(je>~CPkrYtnah0lD zg?)U`QI7}w6MQG78LD!@+;eepk?3k~7gZx_*|)07B~*Nuzkn%Q@1hQ-qoXqehN<(& zs*_+tB;ODWFS>L4Z5*sq6Wzng$$JW;OFTTm|9VD1_Mv>ckQoARU% z=yeAV3Q?&BDHP9&;{kc42?L#3J4DzmIptPBb&so<3WJ+EKSyr&Kpi#sljumfJvB~D z5qd~%dw;tGcgrWEg10@!R@hCJ+NqV-)*5BlF%G6`znSZtH1f2<5DORE4d7$tCKb2DA@uK%uCRX_<#m6iV{}RUSLaM;RM!t@rwXJ zK`if~w;N490c>j{>CnWfV+KI)4$V)44!p@S4bUe!m|a7sL@r&rbp6Zn_R-tj%4+9N zrc7USvWz?wKOemZrbwyx8ul%ngRv1U9iigivprp6_!eU@wzOkY-T<$IPW~!Q0%);^ z1_4pG*qfep4y?aZ<}Ophiz_=*_W6W#hA8I-v1EwN43o8PQ}m(WDw>9Zbxh|iAw>k1 z=BxaO>4lL+GT~hIZ}CMR1s>%P7%~C7?lE1evIyK9>-Lxty{~W ziE$(Q9SdJAE(q*w0z2_;P>&V~cH@3vR`6jg*lg8O+6wI30Y5SOV!q;OD>3`vvh%o2 zuEW>nb2r?d%;in$3d&IWgLZ1$6j8Ay5^gb8?ic}zif%8YNLn(vg!qy4DYXPp% z{=3%uBw@t0$?a2eC-ZLWtP=yA&w{@R9ib7`#rL&Xq_&City@Sna{Sj|4P&6n$dFbhh^Xdg$-; zgDe+Jq%58(P@^}@`QZJm;F4H*9QDHopNwS+pQwm3H14M? zRMiQ0JoVy|o=X$9lR>1|PAIq5J)E~8>gHA*J(r>{!Ep3;>b?y4g-)8V{f-81*pp*8 z2?S>5(m>}wKLxTjj_EJJbc)(3=`fnDtPAg0P+OPyS;etz;o>^Ai-b8h%w(2A)eofC zmr16@`}Ybs7tvcK4OS1?Lcv3;T75N$b6U@VFE%TozH~O`?S7rR@@yowntCs+6x{g> zbM;FM`PYw4$J)TVUoQqo64SgkVah~QywrEbP7*Ifty z!qJ7FfDymxBYQ|>eZkq`$yq0?^2A$sHHmTViMJq9gr5P}xh|)HEF~dl*Nl2ov#3xN z6!x%3Fp3*2i;TfnQTh!6~d3KAq6%~lL_jntUsEqg!(ZxBVboCkJ7 z9OfIKT+Jme^5DnVQOvTppS4w$j%-I0xpuX_BEpcEr33C)drJ z)v5)C&{j=hP8sGKu*Lx#jSN52BZ}xD@u36G<`TY+C+0t6XjtftwQ?HKn@uUx9#qrK zUtAs%oLhW@JL?Zx-az7|Nb@dwref7y3a1=9>L$*8)HP%VJQl}I+ zA-$NonZ&sy-eWQ0v^iac3z&rnz{@nF@gRGd9uygYw;#$3_XUyM!B^+Fsx!|d5zcNyW2~@fQXjvavF$uDVVk?+%%$rZi zaD}&idGh4VGd2S>oA+%tMTEsADEyB4VwjrvxO>AD2MqmyQ^hbXs{h_`Px)1u%T`E= zKAM|$K2W0N?v2=$vPBdB+Vf9&S4qE#^TqmSb{RAym9g8v@r54U`Lf5YQiuF7k4)Mh zN+g!<5sc-I&hFc?5MJq@JApqiZ$>q~DuXy*#d9C%K<|S>AqsO8FO6-_{=0&VO12OX za-2W=*ppC2Cs_ldoyKORY5uv$&2~j-9x0~cHFgsiQXj!w_+8Q1$*&;s=7ynFI^dw2 zND1Fml{r|)SONgP0_EHOW5h7|!53a?3I>Bidr(Pi%j<9*raso|PmNtdawZCg1)|ie z0ygsU->38nfQL}o>Bb>t`#R>(=T2oV`9-zk0)qnr(24$M#kvn%quPKGk^AGOW>v59 zE4e=;Biq!x)&;susb`VxDPGY7IY8BD=pZndz-o)G3x%4Po0}uvDl7W14BKX5hP{W= zQ?q*XyX#M9a4>?`kdx;kzyA1~zLLb}hw{r7$&-_pWf}15mmx)ha7Lwgn${sDsa)B& zAiI(V%a;~r1~fgTbS{f@3attI8drEY@62v9Vxmf|w>sRLeWoh3CJ*X)VCNo{W(sINFx#8# zo1L549_LAni#pJ{-~m9V1fuLg9v2|b+UE+&*_M(M;RyzSa6R%bXCzb^W~$r|TgF(aAAO#aSzN1V<@$zJn;eA zwtcGb`f)PcS{~G4tnDbSjM7s-Cnn7ze=_Kofv=Kuq`Cci^7ofeWFnfd40o;c(+cVb z{D~iJTWkh)z$eTIXSD zW9~)m%qc@#r+M~s3aHmLT~rEy6)Q2asS7f@wJoH&KF5aBLfmhL!c~ng_LXDa?tg8$ z+gi>LRlvlkP6#RRObMM}G7K$o@dgKpgJXKg_>VE`&*9PUKdL;gZzCqO;XFwMsHc`> zOod3At{4as50h&saNR~YG#4_}2p7tuSPFa|G!;FGwX&273J!S)rgbNWs$oGJLr zsTr0~`uMyCyUGdS?r)5a;h5qHaqg>a;&pKmwr{~M;qZ#<`_Bv5D?Mlit`K7TIWjnw z8WEa|;9RmW1tzbf=P84~3mlwnVDGdJo0zo=?_1V@;>7G$Z-&&V8p>oES%M8l;pT5S z#6wJz18c$6T(NT?+f6=>$y})0)r{QHo^UTpY}euEa^1M!GX#G{cQ1Qj>MNK_NBTudYqH4YH=sA zxS~2E%@^0OdOO5$1N)Zo^yz3BB+34dp7oau(f1#(Db`yXru#-cS}&ie(QAJp$wFkt zUJ3Qh%yL;4a$o3X5*4}m`a%RSXh22ve=4DocOe^dP5nAtIe-_sNW_B2*IZ|4MjO3I z6ZtWbeoZeqH+QN2)vH%X&{h7p==-v@T;-0CgdF^;iu&eKPQKaaXQ+zGQGqv4?`BLV zq?nplwl{c14}bXNBIJ;Xah+j9ee=D-{XsrjZCEcMKkz!-X=8J9;?Y}~qR>*>`KvH% zYFf5uq84u~m$=}U;2ggd+c9?w5iW`GL6Sdqt_F|2HaZnOS|Rj^ULY(tN~?27_H3b;7=X!#ntz0;C{BHGR=ZQ;(HnZ zgwcH@T|!`|C?KMJ7)`!(9lyv}j!XNqM$o?xQ4+)m|k<`5uM)WIgc z8=IImr%yA9<_J~UQW~mzDxm}v`nZ8MK?NLi_;~&Ro!K>w%@IFxTrfpq+S(@RnNx4< z;ZxKyKFzMpkD75E++CO)&~ov8!{wfRt6pI7$F2u8l%I|J1)zoKy`AUP#Lzmtwn9fk2{1Uv&tY?79B}7deF>|v2_V+L#W(Rt_i?xUpqs{&(w^Lc-86rJP(0D z$R{R!Gfi&%LLblqUV=d6z}tM5riO3oAPraf>J|54D(e~GbtCHkt_&z`^eNu6cju^) z%q2Q`M0o70Ow)_@TlRm(FR)aSe{L9py~Vm=DfMx-+#)K^O`9ok)#|9`(*Sf$xi}pd zS5$m@Y2{h#<)sa8hm^K)muM3@8-NA%N`~yLsH%16vAnx*TRr1Kp+wvbj(p{1|4;X^jW;!J+Y*rH@4I^V}p; z7GHp9J+}Pl5{fMQA-#74q9$dvQPM*3z^w4@B7pX=JElF5?*Y^AsjJELgXw3`O0KRx z5CsiXf&J00iY*W5?nEPPyYQ_8gnCW_93!)fm`FY@r&1CVvx@jrftC5KK5Z^y&)&4% zG-kT_V5>0r(tSTZTrJOy4p3Xl^Tdz~9Ug4tz#9`mxg1LJAj?BIyKca2Dvg3reNz zk3_1jbe|+Lv*?IS(Gc{ZkcLUq{%FhAZYb&MiX{W1#u-7jA|u-ApFRi#<{a&+B5Xf$ z+F_tRP9blfz9#Ou`jz04$C%DeN!?skBJYUlHhRD`vrs!Fj*RGZaY+fQD`v2R(Lhp) zIij9@(IL-BBQ^9Kz-^VY!=rx}vwTNfe{kzRnxkWrm+I?_Qv66*8rWlqa~r2&0cL51#s-jBsYDwQf;3m)sB}{VrZaT z%5P-&-T-gZ%}4a8%m&c)wxdFy{=gn~vlG(M(+8_nEEjOZ3p}9z@@OVKd#-MtV-?ar zP;l5(S+y_yM8>X;u((8LFcGSI=s&HCGNyE^dUEjP@=e8>{MP!xJncFJZUw1+{Ta(M zH#S&f*@!mv2T@?OT1edkKJA`)KHC5HbIsgLSna)dJsJOsq9nsoSUuJVVkQJAK#-C} z0Yj^}wZZq7hV7Sq4kHqWE1U5~imfC7lScFpllH_C`NnChF;`84-Wp&fUH~bA4)*aV zUcK`_tbng1+ey&pFPWK{cQu7No?{- zZu^GyeDqFB)vI`J(l}+*jCs`!o|`vsdWjlFhF4rA8EUiwFxj$Wkv0GL;nDm}qNi8I zjBfqE8Z?+12!OHKwYPH5!1^y-e1DyYT82x$ZKb?z6yF8el*Q}Bc($3!0PFGOSXv2z z%)-jZT?T|Y@QElCUnY?F%%Ei(pa61H3;JevjtAZRbEhP&NP$Mat0@GU>N%2wsnU75 zhT3^Kdvk618;K8f@DDNP1;{y=8sm10u3CBzseHvfOse@7ezx}AyLTQs zJ1b*ZCt|3+zIjxI06YDy-r^}U@b{Gn*>zc5|EdrT2ldtLkp1}OoD zyZZQWSA2`SbdqocPyqY#XDwGFL#n3I$0+wd^NbjB9aEK#|FkO-PhXZjgfIkNm79`)Qf($lDMfpxtsfp{sxE~pFn50@ijwdeP0uE` z-?kZ_Jr8B*`HX1CQb_{S+f<

    v!^jEB+tTDVym_R{ky9Ar2%^CHZ<`$7I{|TmC_n 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℘G72vJ(S~fGX*;?Q`Qi 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 literal 0 HcmV?d00001 diff --git a/kubespray/docs/img/kubernetes-logo.png b/kubespray/docs/img/kubernetes-logo.png new file mode 100644 index 0000000000000000000000000000000000000000..2838a1829ff3588df5e0d8110a62703031be03ee GIT binary patch literal 6954 zcmZ`;bx;&e*uJAv?r4rakWT5Yqa+0B5ad8YLXnW>=%Y)eLqb4N8fi{A`al$ypf?cB^iVa002cAe~$=kawOhR-wI^1;r zxf28~Y6fZmKwT>NwFBXu&f%!@*Z=?s;sF2<(Ez~Zoe8lC0QiXk0NYOi0QqbHfZ=6+ zo3Y~E1QA?M8+NzkT`TUa$h@-<`|28K5@QKKWK0TlPq^~|0Emb#OwBZ4Zm-~VzUf-) zK=(#Ize0n;e3qu=9Ny2jJkml>UvN)Z4kq_k8f$pv+I5I)*68Yc=~uVL_k#@J6f)w? z+zCK(2$(-1GLpl9-tuX1&ds7qp^v}ISpICz>=Vn*vHhc2DvaDl==b@;jnM62$xHA= zS2MZr2Vn7g_k@=BM#hw_AKYuV7dii33@iQ}EX^9fEX{Bu_kZtaVMc%fD&xNa+90|C z?g$dZFV8)OWIFk7a>n%i`j=5(5<% z;|qGQqC8xW7YY=@Dwr-tcMox zmo9X)3#)6mckmJc;P?Ufb)<^qyA58||BFO$=^EPCG8+|Ri?)0^_UkJm9uZ<@vN2=Z(^WhNzH4tFuZfnGq`0q4!ngEQyIeu@1bun2B z)r{15@(Avh_%Sj(7kQZ(p$#rjcHtuo&V@&pCS0ytzW6vE!4<)k77(%XL9{ZsCRG4I z6vEts7q4=Vw{JyuT~gHF(EXChSl7UXI*UZ!vYj3tJSTia)c zutN_)at5ioR>PzXL75oKmdbSPo7sM1KQQ)T88S~)XuYi=u~I}z*3%le)YTaM7cn-^ z9!nZ)GD%LEt)_Axx|Xgj>hP5+wB(mW-b_uqs=-$k+9bX|fo*T;m^EyDoDuByW?Ck} z`E%byBn{BV+F8vFyL%kTp56%|5Q`!^0#z5CI*%u5cgy|9l!phN@zOo0oP2td zufEP=tZEkjEiSlZp|vIFEpsmnJ{&)sY6?=d^|^a1+UyQoVP>TIwJK%6^h#{GAE!Sn z()+vLT9UGUg4ZnT+LU(QOf6UB`^37jovDMM=n!&p^b&i#z^T-L)v>xlp1)?RXBhw6 zK6x&p))_yJe$Q}M_{0}J5w5*cyF|oe)oZ!XP0;#xz}xW6=0_&s;dQEv!mgUKgDbsY?jM-^}nu9(D@etDfr%*4rGwOw6d zq%TIhtQ^_*G@r=BkwW4FB3`F;eb+M2#HfwQC(AxtWp<-oJMS*oo2Fxu75Njsz2&9R z>po?>{Np7vCG&+i-sbm%9>R z*FD7qJI!dmh0V884m&w(E|lNIyCWpo0n%;C6g)NCR z!S077^CzB8+SlbMs+TFFvXB;v0N}Yut+d7yHg5498 z(u`b(6zxa;Ic_L^5{=me%vLJ+!+N>PB)}xy=tw3V@%b6kuq(-raE2TcUfjY8YHLvO zhE`XGDxKALIZ5)&9MJ7sIr5@w$DQQIyf}qJECYx8W-v-6*tM5$IZj6`UvyL=4xg8k zdBQ*kV0EVcn2&O0E+vz}LaeN38O@*Z<+jI0jb7yW>c7r>*jO=QCS#d&g*f7o_ww(rszJd>eDf?I3~?^lIIcH{A7>4-#}9eCkp0^z z+A7Jf<;*JIEX8+**rPFwd;41I7?mtZ$wT4exLJFmnqo2geP!X$UU#}o11OE~XCzYYTf8MT>K1CS_|B6^omT@K*jL^?62NHHtn}Hi5=4Pp;k1%=#=h>@q zfMQJLgD{wZT+XUo0V4m+iDz3*`AEo=@|gkh`87SAyt)_*fHDJNW7=kwy%}6Z#h$5UfX~R7DU>VJq@iLRk|8v zWC0<-044`@NIh5h+eyte^)b71r6Kc0J)O~U4-Nlz*po8{n7pOVU#vVn=kjN!Ify2D zzHj|j%E^IBy@Mv_Koln0jI1Cyr)nsO#|AUO_iJfxSyowwU}zBdqI;8PDh~7d-y5e` zS#smoc9`)o#vZ4em6xI!;}(=dS#``4XFwTfzm}fCIb}U*X#8qZK;dlI2w=0%iY$yw zTz&4>5E;5~x+!97T+?GhHxDbw-$uxm_!JinU6zJTs`$jF3O*7pYc%36_#Diocx=lo z`i1=OvXAE^Yg(Au+f7^XiZmJ?2C+)hf${ee)vjA<^aLRgt@VH^CdlyH!>ZjCP5NJW zk578;lX$Pet&}Y{RjV(jR{4wBz=!s)0>Yp}iNXj_O+l+t>gI8LY?Z&$`%cH;W}=~H z4AJMoF`A0`m?pydhu_V)_kp9xlW{1qz&Y{c& zLz{1f7^`E=4)~jb$c222bl`4Cv$dNa2iuU#cZa(y%^zAeu zgNl0(08O(Xm?E3OByqO>=@ur_=gSW{@=EXu>JejI>#}RWf*w2vGEL=z@85-2Gia`N zy^wWb+R?9@W7|D|-h|!~S2b~&;XJufv-SjXG&!zU@IY`2aDz|=q~mhnt^pr;#G6w) z0vf3+$0eg?QD1Ytrn6Z0(>p{KoTRCFn!l>l5qlTSnoS5qyV-oy;U6RV?zi6U&x}xK zjQU+E4P@HaCz|!Rl+kSdG!rnG*pU6w1kH7yMK+AdtMz^U$QE- z`lopfjC214#$n!t1r2=b z*K!i)4pLX-9-yVrCh1LK2BlAu{<=3!L}kR@E6eLo#{qi|4!=K!J+LY}iroxK$&R(iK7zo;poqL8H_UEb+_>|(2Ap`ZAQ{no88J)tPf-}CAeM(j-B%S@b zk-ggj*I595SG`p>Q*)H#xH^Xu{d!cfw%15Z_pKI6Z8piaz8;CYrll_v_+zl=U1-l* zRwo?$-aX(qB&?>N;gKjdK`sVXlF&DzFw-V8JZY_qU#L9SJ16WQy^AdNqbksHW_Pm} zhx`z)A+X!iywE5>lvM;)JUqJXdRlx>#n@`GK_aTFqo%vRx_d0Wc6uV7k!2@h2PE~K z^ewW&S%A8+3D_1N&JMqR8MV91zL$E(Ny#vHLctZ>H=5S&YyUV|ROSa-F}LWvv2X!- z^@lxSe#x^mou;(pD=+T6;Br5;byi&INAy$ErbH%3MWIjyzMM#~CaOl9|{+X>6qNFX1ZM# zN9dVZzcg`(Gd|*S_hW+&DG&;%QCj^Um5=t_x2NfF|JSm=M=n>t)tpk2QUJQ>^i8w3?Do@P~NmEJph ztYBH#dJv$(FnU_*o^f=$usz02-mMfdh-O0zXAOPNC#T?cSQX61rct^;6Z?mHW|JCj z!|AIMu}{~+Qngz;S?6_|ttL6K4p3hhyO0*6hIBalYpe=F$HzZEWk$xj;ksdf%UAyR zx4f$~O><^jAR{Vf@LqvuUVS>upHl?=owE^6r7!qR^3M(+mt4|YVL|`Z<}+$Zm~HrG ztNGN6R3o-=s{n6t1~&V_S&7iQzu~FY$8PUc(PUD8d@9**+0hZDqi@VMF1PQ>Zc${6 zwiwFUDr@y#EYa%m10 z8G~Q1Aa9q3(&l4ZE^|Gn{V+%^2Gn+Y+~DhT$wEzU;sdUqNQpKgg>A=cS8s{0cGf-T z4qjfZWeW1IF51=1y(r$l@r%yF#xxcDmtMfsv5DmF4V1fYw!w%NTU%mI6redj109i; zH#UUNM>~o4_-Z!5@U~v*;xKe=J-xAl5G*amUwJ6{Ni@N8jJ9lDM9b*q&jT{<71y|n zMFFQDa!i5S;4{OL2)@V=;uYHrhVB%ddkv#EovC8hqHsP)%6SzP2h>rT{{wz9FBpGj zrFo|T{OL>py}yJ6J&{vC$6=IDMSb#jivl1i&btUYOLnq`A9 zB6p6UOItLZ3!(P~#y?3FOL@wrFPNUQgKShD2svvf@p-Pj-bUdda@rCr>s)Lyjb|Uc z1iH%>C2}1uJA!U&aBQD;;8ZOqKX~XdIhzADND4#HFpdh{YSKr(DD!S#7u1O7D(XB4W&1Vvt?zl;VYDUiH zR<~gRr!RYGb}hz%*|I-BHFTSfB)Kq)39h!0tH_WaD+N(tqTRAChpw7jCoB~Q^(^1f zKHP@=r|~Z~{Zh}2mRc=FRo;y5Cwf#&8f_x<|K{1<*T1+AXIt;6@iSWBoNbeRw94pO zEv7-Zl!3h2W|`bkmkx7ImM<%y(62S%c)cR^k5t!++{!mHq|^S{xD%Dtb?xTyqKEnf z4vS~YyBXxrsYNoF?WiUtzt;877aK;-#l>j_{(L3FYt(tg=>p=G0YtM%k+G6ODB7`c z`1*HzzMxU86K?InemRmhEg<0+2qvI6P0WiYfZ!aree<;uj`!%~tB);*V14u;r5v;Ei5I2VsUypO8V z*P3L~kAD}<{0SaC71FTPrY}_WU)bj$D)un+P3BDMAL^%murqQ$&Oxvqr-JNFHz1vf z;tumr-KCx-x|rd@P}j`QZ$X3mOI!j+gR&+X45y<8S9sZ>KGOVg(v&@NjbHn!1*65j z7uvQ>@oUD<#Dv6pDM5iw?RjoxS~e*tIk*ICZW^Mv2<}CZk|<~m>Zd#QP-R4Mm62{u zO5Ra;(fhYVUzLCU7wNUwLX=Yp4q7OZUG?EWd5r`e{kk31r1l!&jgr~0u+wRJ>{fgm zU5nX7lLgkhZ;tq~V@Hr%V7$v{k;gBCu7^`kx1d;-TDfi1e@_?oLV@yPcw9Dm#GQnb z7z|%}_U?D|%jMBBY#@w5k-BbuI@^{*bytXet>oWiLBI*m2)Oe3k;>(pb0qTa_9_~ z9=L(|>?UrF8I$ClghW?ohNlNT{{ewX6-6M*WeXpBoM9jF`FqE_1m~r~Sgv z-YAYJm)h_|!lT4Q@t*c%TkF-_t5fN|mYW-*-1$CuO?6eh9XxauUv6sWkNZfDV$W!g zW^%{1*P!lkDl~%pH6T5V_N<_!JBVhe{25;<=lkp_;5%`jag9|rk?jWqU1@)!gg@6# zI6I{~W%A6^2;rMBmDp>6drdMgQf^4R|H-0~L18ho8;m_~@LrISAK{3HK7cU+L|4exM;rs=wmK>2>WxZa$ znQE0iFN?8aLrFv8!Dp~HTo5u2eyrGD9|exWGksI`Hsv;&(c?1n;Is2tT(^=5E15W1 z^D|=$9$32OOR-d;eVBh^0r4R#J24ETTi%_~LmOI@cMt6~Ilucu7HY71Jye_Y{A*yX zi(H9leo?^py`PM>&b42S>ksN${h9vo{GGRRdtoB0eCl0!gvlyBdk7%SwK2wf|CiHB z>FY4>=Y?T3wMO5V(SGK?GnY<|260cJasSNTmv%;SQyrg8zluT?kQI(2q^Kw+1Hpl- zhg)ifSEosYe&}Zt%D_XPdqYk4h@;a}4HSt8j)KD20$Cud`Yj)T;|c7$sZo;cW1-(^ z{ + ``` + +7. If you have several temporary-stage commits - squash them using [git rebase -i](https://eli.thegreenplace.net/2014/02/19/squashing-github-pull-requests-into-a-single-commit) + + Also you could use interactive rebase + + ```ShellSession + git rebase -i HEAD~10 + ``` + + to delete commits which you don't want to contribute into original repo. + +8. When your changes is in place, you need to check upstream repo one more time because it could be changed during your work. + + Check that you're on correct branch: + + ```ShellSession + git status + ``` + + And pull changes from upstream (if any): + + ```ShellSession + git pull --rebase upstream master + ``` + +9. Now push your changes to your **fork** repo with + + ```ShellSession + git push + ``` + + If your branch doesn't exist on github, git will propose you to use something like + + ```ShellSession + git push --set-upstream origin fixes-name-date-index + ``` + +10. Open you forked repo in browser, on the main page you will see proposition to create pull request for your newly created branch. Check proposed diff of your PR. If something is wrong you could safely delete "fix" branch on github using + + ```ShellSession + git push origin --delete fixes-name-date-index + git branch -D fixes-name-date-index + ``` + + and start whole process from the beginning. + + If everything is fine - add description about your changes (what they do and why they're needed) and confirm pull request creation. diff --git a/kubespray/docs/kata-containers.md b/kubespray/docs/kata-containers.md new file mode 100644 index 0000000..30843fd --- /dev/null +++ b/kubespray/docs/kata-containers.md @@ -0,0 +1,101 @@ +# Kata Containers + +[Kata Containers](https://katacontainers.io) is a secure container runtime with lightweight virtual machines that supports multiple hypervisor solutions. + +## Hypervisors + +_Qemu_ is the only hypervisor supported by Kubespray. + +## Installation + +To enable Kata Containers, set the following variables: + +**k8s-cluster.yml**: + +```yaml +container_manager: containerd +kata_containers_enabled: true +``` + +**etcd.yml**: + +```yaml +etcd_deployment_type: host +``` + +## Usage + +By default, runc is used for pods. +Kubespray generates the runtimeClass kata-qemu, and it is necessary to specify it as +the runtimeClassName of a pod spec to use Kata Containers: + +```shell +$ kubectl get runtimeclass +NAME HANDLER AGE +kata-qemu kata-qemu 3m34s +$ +$ cat nginx.yaml +apiVersion: v1 +kind: Pod +metadata: + name: mypod +spec: + runtimeClassName: kata-qemu + containers: + - name: nginx + image: nginx:1.14.2 +$ +$ kubectl apply -f nginx.yaml +``` + +## Configuration + +### Recommended : Pod Overhead + +[Pod Overhead](https://kubernetes.io/docs/concepts/configuration/pod-overhead/) is a feature for accounting for the resources consumed by the Runtime Class used by the Pod. + +When this feature is enabled, Kubernetes will count the fixed amount of CPU and memory set in the configuration as used by the virtual machine and not by the containers running in the Pod. + +Pod Overhead is mandatory if you run Pods with Kata Containers that use [resources limits](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/#requests-and-limits). + +**Set cgroup driver**: + +To enable Pod Overhead feature you have to configure Kubelet with the appropriate cgroup driver, using the following configuration: + +`cgroupfs` works best: + +```yaml +kubelet_cgroup_driver: cgroupfs +``` + +... but when using `cgroups v2` (see ) you can use systemd as well: + +```yaml +kubelet_cgroup_driver: systemd +``` + +**Qemu hypervisor configuration**: + +The configuration for the Qemu hypervisor uses the following values: + +```yaml +kata_containers_qemu_overhead: true +kata_containers_qemu_overhead_fixed_cpu: 10m +kata_containers_qemu_overhead_fixed_memory: 290Mi +``` + +### Optional : Select Kata Containers version + +Optionally you can select the Kata Containers release version to be installed. The available releases are published in [GitHub](https://github.com/kata-containers/kata-containers/releases). + +```yaml +kata_containers_version: 2.2.2 +``` + +### Optional : Debug + +Debug is disabled by default for all the components of Kata Containers. You can change this behaviour with the following configuration: + +```yaml +kata_containers_qemu_debug: 'false' +``` diff --git a/kubespray/docs/kube-ovn.md b/kubespray/docs/kube-ovn.md new file mode 100644 index 0000000..26d7cd9 --- /dev/null +++ b/kubespray/docs/kube-ovn.md @@ -0,0 +1,55 @@ +# Kube-OVN + +Kube-OVN integrates the OVN-based Network Virtualization with Kubernetes. It offers an advanced Container Network Fabric for Enterprises. + +For more information please check [Kube-OVN documentation](https://github.com/alauda/kube-ovn) + +**Warning:** Kernel version (`cat /proc/version`) needs to be different from `3.10.0-862` or kube-ovn won't start and will print this message: + +```bash +kernel version 3.10.0-862 has a nat related bug that will affect ovs function, please update to a version greater than 3.10.0-898 +``` + +## How to use it + +Enable kube-ovn in `group_vars/k8s_cluster/k8s_cluster.yml` + +```yml +... +kube_network_plugin: kube-ovn +... +``` + +## Verifying kube-ovn install + +Kube-OVN run ovn and controller in `kube-ovn` namespace + +* Check the status of kube-ovn pods + +```ShellSession +# From the CLI +kubectl get pod -n kube-ovn + +# Output +NAME READY STATUS RESTARTS AGE +kube-ovn-cni-49lsm 1/1 Running 0 2d20h +kube-ovn-cni-9db8f 1/1 Running 0 2d20h +kube-ovn-cni-wftdk 1/1 Running 0 2d20h +kube-ovn-controller-68d7bb48bd-7tnvg 1/1 Running 0 2d21h +ovn-central-6675dbb7d9-d7z8m 1/1 Running 0 4d16h +ovs-ovn-hqn8p 1/1 Running 0 4d16h +ovs-ovn-hvpl8 1/1 Running 0 4d16h +ovs-ovn-r5frh 1/1 Running 0 4d16h +``` + +* Check the default and node subnet + +```ShellSession +# From the CLI +kubectl get subnet + +# Output +NAME PROTOCOL CIDR PRIVATE NAT +join IPv4 100.64.0.0/16 false false +ovn-default IPv4 10.16.0.0/16 false true +``` diff --git a/kubespray/docs/kube-router.md b/kubespray/docs/kube-router.md new file mode 100644 index 0000000..54e366c --- /dev/null +++ b/kubespray/docs/kube-router.md @@ -0,0 +1,79 @@ +# Kube-router + +Kube-router is a L3 CNI provider, as such it will setup IPv4 routing between +nodes to provide Pods' networks reachability. + +See [kube-router documentation](https://www.kube-router.io/). + +## Verifying kube-router install + +Kube-router runs its pods as a `DaemonSet` in the `kube-system` namespace: + +* Check the status of kube-router pods + +```ShellSession +# From the CLI +kubectl get pod --namespace=kube-system -l k8s-app=kube-router -owide + +# output +NAME READY STATUS RESTARTS AGE IP NODE NOMINATED NODE +kube-router-4f679 1/1 Running 0 2d 192.168.186.4 mykube-k8s-node-nf-2 +kube-router-5slf8 1/1 Running 0 2d 192.168.186.11 mykube-k8s-node-nf-3 +kube-router-lb6k2 1/1 Running 0 20h 192.168.186.14 mykube-k8s-node-nf-6 +kube-router-rzvrb 1/1 Running 0 20h 192.168.186.17 mykube-k8s-node-nf-4 +kube-router-v6n56 1/1 Running 0 2d 192.168.186.6 mykube-k8s-node-nf-1 +kube-router-wwhg8 1/1 Running 0 20h 192.168.186.16 mykube-k8s-node-nf-5 +kube-router-x2xs7 1/1 Running 0 2d 192.168.186.10 mykube-k8s-master-1 +``` + +* Peek at kube-router container logs: + +```ShellSession +# From the CLI +kubectl logs --namespace=kube-system -l k8s-app=kube-router | grep Peer.Up + +# output +time="2018-09-17T16:47:14Z" level=info msg="Peer Up" Key=192.168.186.6 State=BGP_FSM_OPENCONFIRM Topic=Peer +time="2018-09-17T16:47:16Z" level=info msg="Peer Up" Key=192.168.186.11 State=BGP_FSM_OPENCONFIRM Topic=Peer +time="2018-09-17T16:47:46Z" level=info msg="Peer Up" Key=192.168.186.10 State=BGP_FSM_OPENCONFIRM Topic=Peer +time="2018-09-18T19:12:24Z" level=info msg="Peer Up" Key=192.168.186.14 State=BGP_FSM_OPENCONFIRM Topic=Peer +time="2018-09-18T19:12:28Z" level=info msg="Peer Up" Key=192.168.186.17 State=BGP_FSM_OPENCONFIRM Topic=Peer +time="2018-09-18T19:12:38Z" level=info msg="Peer Up" Key=192.168.186.16 State=BGP_FSM_OPENCONFIRM Topic=Peer +[...] +``` + +## Gathering kube-router state + +Kube-router Pods come bundled with a "Pod Toolbox" which provides very +useful internal state views for: + +* IPVS: via `ipvsadm` +* BGP peering and routing info: via `gobgp` + +You need to `kubectl exec -it ...` into a kube-router container to use these, see + for details. + +## Kube-router configuration + +You can change the default configuration by overriding `kube_router_...` variables +(as found at `roles/network_plugin/kube-router/defaults/main.yml`), +these are named to follow `kube-router` command-line options as per +. + +## Advanced BGP Capabilities + + + +If you have other networking devices or SDN systems that talk BGP, kube-router will fit in perfectly. +From a simple full node-to-node mesh to per-node peering configurations, most routing needs can be attained. +The configuration is Kubernetes native (annotations) just like the rest of kube-router. + +For more details please refer to the . + +Next options will set up annotations for kube-router, using `kubectl annotate` command. + +```yml +kube_router_annotations_master: [] +kube_router_annotations_node: [] +kube_router_annotations_all: [] +``` diff --git a/kubespray/docs/kube-vip.md b/kubespray/docs/kube-vip.md new file mode 100644 index 0000000..846ec09 --- /dev/null +++ b/kubespray/docs/kube-vip.md @@ -0,0 +1,72 @@ +# kube-vip + +kube-vip provides Kubernetes clusters with a virtual IP and load balancer for both the control plane (for building a highly-available cluster) and Kubernetes Services of type LoadBalancer without relying on any external hardware or software. + +## Prerequisites + +You have to configure `kube_proxy_strict_arp` when the kube_proxy_mode is `ipvs` and kube-vip ARP is enabled. + +```yaml +kube_proxy_strict_arp: true +``` + +## Install + +You have to explicitly enable the kube-vip extension: + +```yaml +kube_vip_enabled: true +``` + +You also need to enable +[kube-vip as HA, Load Balancer, or both](https://kube-vip.io/docs/installation/static/#kube-vip-as-ha-load-balancer-or-both): + +```yaml +# HA for control-plane, requires a VIP +kube_vip_controlplane_enabled: true +kube_vip_address: 10.42.42.42 +loadbalancer_apiserver: + address: "{{ kube_vip_address }}" + port: 6443 +# kube_vip_interface: ens160 + +# LoadBalancer for services +kube_vip_services_enabled: false +# kube_vip_services_interface: ens320 +``` + +> Note: When using `kube-vip` as LoadBalancer for services, +[additional manual steps](https://kube-vip.io/docs/usage/cloud-provider/) +are needed. + +If using [local traffic policy](https://kube-vip.io/docs/usage/kubernetes-services/#external-traffic-policy-kube-vip-v050): + +```yaml +kube_vip_enableServicesElection: true +``` + +If using [ARP mode](https://kube-vip.io/docs/installation/static/#arp) : + +```yaml +kube_vip_arp_enabled: true +``` + +If using [BGP mode](https://kube-vip.io/docs/installation/static/#bgp) : + +```yaml +kube_vip_bgp_enabled: true +kube_vip_local_as: 65000 +kube_vip_bgp_routerid: 192.168.0.2 +kube_vip_bgppeers: +- 192.168.0.10:65000::false +- 192.168.0.11:65000::false +# kube_vip_bgp_peeraddress: +# kube_vip_bgp_peerpass: +# kube_vip_bgp_peeras: +``` + +If using [control plane load-balancing](https://kube-vip.io/docs/about/architecture/#control-plane-load-balancing): + +```yaml +kube_vip_lb_enable: true +``` diff --git a/kubespray/docs/kubernetes-apps/cephfs_provisioner.md b/kubespray/docs/kubernetes-apps/cephfs_provisioner.md new file mode 100644 index 0000000..c5c18db --- /dev/null +++ b/kubespray/docs/kubernetes-apps/cephfs_provisioner.md @@ -0,0 +1,73 @@ +# CephFS Volume Provisioner for Kubernetes 1.5+ + +[![Docker Repository on Quay](https://quay.io/repository/external_storage/cephfs-provisioner/status "Docker Repository on Quay")](https://quay.io/repository/external_storage/cephfs-provisioner) + +Using Ceph volume client + +## Development + +Compile the provisioner + +``` console +make +``` + +Make the container image and push to the registry + +``` console +make push +``` + +## Test instruction + +- Start Kubernetes local cluster + +See [Kubernetes](https://kubernetes.io/) + +- Create a Ceph admin secret + +``` bash +ceph auth get client.admin 2>&1 |grep "key = " |awk '{print $3'} |xargs echo -n > /tmp/secret +kubectl create ns cephfs +kubectl create secret generic ceph-secret-admin --from-file=/tmp/secret --namespace=cephfs +``` + +- Start CephFS provisioner + +The following example uses `cephfs-provisioner-1` as the identity for the instance and assumes kubeconfig is at `/root/.kube`. The identity should remain the same if the provisioner restarts. If there are multiple provisioners, each should have a different identity. + +``` bash +docker run -ti -v /root/.kube:/kube -v /var/run/kubernetes:/var/run/kubernetes --privileged --net=host cephfs-provisioner /usr/local/bin/cephfs-provisioner -master=http://127.0.0.1:8080 -kubeconfig=/kube/config -id=cephfs-provisioner-1 +``` + +Alternatively, deploy it in kubernetes, see [deployment](deploy/README.md). + +- Create a CephFS Storage Class + +Replace Ceph monitor's IP in [example class](example/class.yaml) with your own and create storage class: + +``` bash +kubectl create -f example/class.yaml +``` + +- Create a claim + +``` bash +kubectl create -f example/claim.yaml +``` + +- Create a Pod using the claim + +``` bash +kubectl create -f example/test-pod.yaml +``` + +## Known limitations + +- Kernel CephFS doesn't work with SELinux, setting SELinux label in Pod's securityContext will not work. +- Kernel CephFS doesn't support quota or capacity, capacity requested by PVC is not enforced or validated. +- Currently each Ceph user created by the provisioner has `allow r` MDS cap to permit CephFS mount. + +## Acknowledgement + +Inspired by CephFS Manila provisioner and conversation with John Spray diff --git a/kubespray/docs/kubernetes-apps/local_volume_provisioner.md b/kubespray/docs/kubernetes-apps/local_volume_provisioner.md new file mode 100644 index 0000000..e9c6225 --- /dev/null +++ b/kubespray/docs/kubernetes-apps/local_volume_provisioner.md @@ -0,0 +1,131 @@ +# Local Static Storage Provisioner + +The [local static storage provisioner](https://github.com/kubernetes-sigs/sig-storage-local-static-provisioner) +is NOT a dynamic storage provisioner as you would +expect from a cloud provider. Instead, it simply creates PersistentVolumes for +all mounts under the `host_dir` of the specified storage class. +These storage classes are specified in the `local_volume_provisioner_storage_classes` nested dictionary. + +Example: + +```yaml +local_volume_provisioner_storage_classes: + local-storage: + host_dir: /mnt/disks + mount_dir: /mnt/disks + fast-disks: + host_dir: /mnt/fast-disks + mount_dir: /mnt/fast-disks + block_cleaner_command: + - "/scripts/shred.sh" + - "2" + volume_mode: Filesystem + fs_type: ext4 +``` + +For each key in `local_volume_provisioner_storage_classes` a "storage class" with +the same name is created in the entry `storageClassMap` of the ConfigMap `local-volume-provisioner`. +The subkeys of each storage class in `local_volume_provisioner_storage_classes` +are converted to camelCase and added as attributes to the storage class in the +ConfigMap. + +The result of the above example is: + +```yaml +data: + storageClassMap: | + local-storage: + hostDir: /mnt/disks + mountDir: /mnt/disks + fast-disks: + hostDir: /mnt/fast-disks + mountDir: /mnt/fast-disks + blockCleanerCommand: + - "/scripts/shred.sh" + - "2" + volumeMode: Filesystem + fsType: ext4 +``` + +Additionally, a StorageClass object (`storageclasses.storage.k8s.io`) is also +created for each storage class: + +```bash +$ kubectl get storageclasses.storage.k8s.io +NAME PROVISIONER RECLAIMPOLICY +fast-disks kubernetes.io/no-provisioner Delete +local-storage kubernetes.io/no-provisioner Delete +``` + +The default StorageClass is `local-storage` on `/mnt/disks`; +the rest of this documentation will use that path as an example. + +## Examples to create local storage volumes + +1. Using tmpfs + + ```bash + for vol in vol1 vol2 vol3; do + mkdir /mnt/disks/$vol + mount -t tmpfs -o size=5G $vol /mnt/disks/$vol + done + ``` + + The tmpfs method is not recommended for production because the mounts are not + persistent and data will be deleted on reboot. + +1. Mount physical disks + + ```bash + mkdir /mnt/disks/ssd1 + mount /dev/vdb1 /mnt/disks/ssd1 + ``` + + Physical disks are recommended for production environments because it offers + complete isolation in terms of I/O and capacity. + +1. Mount unpartitioned physical devices + + ```bash + for disk in /dev/sdc /dev/sdd /dev/sde; do + ln -s $disk /mnt/disks + done + ``` + + This saves time of precreating filesystems. Note that your storageclass must have + `volume_mode` set to `"Filesystem"` and `fs_type` defined. If either is not set, the + disk will be added as a raw block device. + +1. PersistentVolumes with `volumeMode="Block"` + + Just like above, you can create PersistentVolumes with volumeMode `Block` + by creating a symbolic link under discovery directory to the block device on + the node, if you set `volume_mode` to `"Block"`. This will create a volume + presented into a Pod as a block device, without any filesystem on it. + +1. File-backed sparsefile method + + ```bash + truncate /mnt/disks/disk5 --size 2G + mkfs.ext4 /mnt/disks/disk5 + mkdir /mnt/disks/vol5 + mount /mnt/disks/disk5 /mnt/disks/vol5 + ``` + + If you have a development environment and only one disk, this is the best way + to limit the quota of persistent volumes. + +1. Simple directories + + In a development environment, using `mount --bind` works also, but there is no capacity + management. + +## Usage notes + +Make sure to make any mounts persist via `/etc/fstab` or with systemd mounts (for +Flatcar Container Linux or Fedora CoreOS). Pods with persistent volume claims will not be +able to start if the mounts become unavailable. + +## Further reading + +Refer to the upstream docs here: diff --git a/kubespray/docs/kubernetes-apps/rbd_provisioner.md b/kubespray/docs/kubernetes-apps/rbd_provisioner.md new file mode 100644 index 0000000..02d3fa3 --- /dev/null +++ b/kubespray/docs/kubernetes-apps/rbd_provisioner.md @@ -0,0 +1,79 @@ +# RBD Volume Provisioner for Kubernetes 1.5+ + +`rbd-provisioner` is an out-of-tree dynamic provisioner for Kubernetes 1.5+. +You can use it quickly & easily deploy ceph RBD storage that works almost +anywhere. + +It works just like in-tree dynamic provisioner. For more information on how +dynamic provisioning works, see [the docs](https://kubernetes.io/docs/concepts/storage/persistent-volumes/) +or [this blog post](http://blog.kubernetes.io/2016/10/dynamic-provisioning-and-storage-in-kubernetes.html). + +## Development + +Compile the provisioner + +```console +make +``` + +Make the container image and push to the registry + +```console +make push +``` + +## Test instruction + +* Start Kubernetes local cluster + +See [Kubernetes](https://kubernetes.io/). + +* Create a Ceph admin secret + +```bash +ceph auth get client.admin 2>&1 |grep "key = " |awk '{print $3'} |xargs echo -n > /tmp/secret +kubectl create secret generic ceph-admin-secret --from-file=/tmp/secret --namespace=kube-system +``` + +* Create a Ceph pool and a user secret + +```bash +ceph osd pool create kube 8 8 +ceph auth add client.kube mon 'allow r' osd 'allow rwx pool=kube' +ceph auth get-key client.kube > /tmp/secret +kubectl create secret generic ceph-secret --from-file=/tmp/secret --namespace=kube-system +``` + +* Start RBD provisioner + +The following example uses `rbd-provisioner-1` as the identity for the instance and assumes kubeconfig is at `/root/.kube`. The identity should remain the same if the provisioner restarts. If there are multiple provisioners, each should have a different identity. + +```bash +docker run -ti -v /root/.kube:/kube -v /var/run/kubernetes:/var/run/kubernetes --privileged --net=host quay.io/external_storage/rbd-provisioner /usr/local/bin/rbd-provisioner -master=http://127.0.0.1:8080 -kubeconfig=/kube/config -id=rbd-provisioner-1 +``` + +Alternatively, deploy it in kubernetes, see [deployment](deploy/README.md). + +* Create a RBD Storage Class + +Replace Ceph monitor's IP in [examples/class.yaml](examples/class.yaml) with your own and create storage class: + +```bash +kubectl create -f examples/class.yaml +``` + +* Create a claim + +```bash +kubectl create -f examples/claim.yaml +``` + +* Create a Pod using the claim + +```bash +kubectl create -f examples/test-pod.yaml +``` + +## Acknowledgements + +* This provisioner is extracted from [Kubernetes core](https://github.com/kubernetes/kubernetes) with some modifications for this project. diff --git a/kubespray/docs/kubernetes-apps/registry.md b/kubespray/docs/kubernetes-apps/registry.md new file mode 100644 index 0000000..182f10a --- /dev/null +++ b/kubespray/docs/kubernetes-apps/registry.md @@ -0,0 +1,244 @@ +# Private Docker Registry in Kubernetes + +Kubernetes offers an optional private Docker registry addon, which you can turn +on when you bring up a cluster or install later. This gives you a place to +store truly private Docker images for your cluster. + +## How it works + +The private registry runs as a `Pod` in your cluster. It does not currently +support SSL or authentication, which triggers Docker's "insecure registry" +logic. To work around this, we run a proxy on each node in the cluster, +exposing a port onto the node (via a hostPort), which Docker accepts as +"secure", since it is accessed by `localhost`. + +## Turning it on + +Some cluster installs (e.g. GCE) support this as a cluster-birth flag. The +`ENABLE_CLUSTER_REGISTRY` variable in `cluster/gce/config-default.sh` governs +whether the registry is run or not. To set this flag, you can specify +`KUBE_ENABLE_CLUSTER_REGISTRY=true` when running `kube-up.sh`. If your cluster +does not include this flag, the following steps should work. Note that some of +this is cloud-provider specific, so you may have to customize it a bit. + +### Make some storage + +The primary job of the registry is to store data. To do that we have to decide +where to store it. For cloud environments that have networked storage, we can +use Kubernetes's `PersistentVolume` abstraction. The following template is +expanded by `salt` in the GCE cluster turnup, but can easily be adapted to +other situations: + +```yaml +kind: PersistentVolume +apiVersion: v1 +metadata: + name: kube-system-kube-registry-pv +spec: +{% if pillar.get('cluster_registry_disk_type', '') == 'gce' %} + capacity: + storage: {{ pillar['cluster_registry_disk_size'] }} + accessModes: + - ReadWriteOnce + gcePersistentDisk: + pdName: "{{ pillar['cluster_registry_disk_name'] }}" + fsType: "ext4" +{% endif %} +``` + +If, for example, you wanted to use NFS you would just need to change the +`gcePersistentDisk` block to `nfs`. See +[here](https://kubernetes.io/docs/concepts/storage/volumes/) for more details on volumes. + +Note that in any case, the storage (in the case the GCE PersistentDisk) must be +created independently - this is not something Kubernetes manages for you (yet). + +### I don't want or don't have persistent storage + +If you are running in a place that doesn't have networked storage, or if you +just want to kick the tires on this without committing to it, you can easily +adapt the `ReplicationController` specification below to use a simple +`emptyDir` volume instead of a `persistentVolumeClaim`. + +## Claim the storage + +Now that the Kubernetes cluster knows that some storage exists, you can put a +claim on that storage. As with the `PersistentVolume` above, you can start +with the `salt` template: + +```yaml +kind: PersistentVolumeClaim +apiVersion: v1 +metadata: + name: kube-registry-pvc + namespace: kube-system +spec: + accessModes: + - ReadWriteOnce + resources: + requests: + storage: {{ pillar['cluster_registry_disk_size'] }} +``` + +This tells Kubernetes that you want to use storage, and the `PersistentVolume` +you created before will be bound to this claim (unless you have other +`PersistentVolumes` in which case those might get bound instead). This claim +gives you the right to use this storage until you release the claim. + +## Run the registry + +Now we can run a Docker registry: + +```yaml +apiVersion: v1 +kind: ReplicationController +metadata: + name: kube-registry-v0 + namespace: kube-system + labels: + k8s-app: registry + version: v0 +spec: + replicas: 1 + selector: + k8s-app: registry + version: v0 + template: + metadata: + labels: + k8s-app: registry + version: v0 + spec: + containers: + - name: registry + image: registry:2 + resources: + limits: + cpu: 100m + memory: 100Mi + env: + - name: REGISTRY_HTTP_ADDR + value: :5000 + - name: REGISTRY_STORAGE_FILESYSTEM_ROOTDIRECTORY + value: /var/lib/registry + volumeMounts: + - name: image-store + mountPath: /var/lib/registry + ports: + - containerPort: 5000 + name: registry + protocol: TCP + volumes: + - name: image-store + persistentVolumeClaim: + claimName: kube-registry-pvc +``` + +*Note:* that if you have set multiple replicas, make sure your CSI driver has support for the `ReadWriteMany` accessMode. + +## Expose the registry in the cluster + +Now that we have a registry `Pod` running, we can expose it as a Service: + +```yaml +apiVersion: v1 +kind: Service +metadata: + name: kube-registry + namespace: kube-system + labels: + k8s-app: registry + kubernetes.io/name: "KubeRegistry" +spec: + selector: + k8s-app: registry + ports: + - name: registry + port: 5000 + protocol: TCP +``` + +## Expose the registry on each node + +Now that we have a running `Service`, we need to expose it onto each Kubernetes +`Node` so that Docker will see it as `localhost`. We can load a `Pod` on every +node by creating following daemonset. + +```yaml +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: kube-registry-proxy + namespace: kube-system + labels: + k8s-app: kube-registry-proxy + version: v0.4 +spec: + template: + metadata: + labels: + k8s-app: kube-registry-proxy + kubernetes.io/name: "kube-registry-proxy" + version: v0.4 + spec: + containers: + - name: kube-registry-proxy + image: gcr.io/google_containers/kube-registry-proxy:0.4 + resources: + limits: + cpu: 100m + memory: 50Mi + env: + - name: REGISTRY_HOST + value: kube-registry.kube-system.svc.cluster.local + - name: REGISTRY_PORT + value: "5000" + ports: + - name: registry + containerPort: 80 + hostPort: 5000 +``` + +When modifying replication-controller, service and daemon-set definitions, take +care to ensure *unique* identifiers for the rc-svc couple and the daemon-set. +Failing to do so will have register the localhost proxy daemon-sets to the +upstream service. As a result they will then try to proxy themselves, which +will, for obvious reasons, not work. + +This ensures that port 5000 on each node is directed to the registry `Service`. +You should be able to verify that it is running by hitting port 5000 with a web +browser and getting a 404 error: + +```ShellSession +$ curl localhost:5000 +404 page not found +``` + +## Using the registry + +To use an image hosted by this registry, simply say this in your `Pod`'s +`spec.containers[].image` field: + +```yaml + image: localhost:5000/user/container +``` + +Before you can use the registry, you have to be able to get images into it, +though. If you are building an image on your Kubernetes `Node`, you can spell +out `localhost:5000` when you build and push. More likely, though, you are +building locally and want to push to your cluster. + +You can use `kubectl` to set up a port-forward from your local node to a +running Pod: + +```ShellSession +$ POD=$(kubectl get pods --namespace kube-system -l k8s-app=registry \ + -o template --template '{{range .items}}{{.metadata.name}} {{.status.phase}}{{"\n"}}{{end}}' \ + | grep Running | head -1 | cut -f1 -d' ') + +$ kubectl port-forward --namespace kube-system $POD 5000:5000 & +``` + +Now you can build and push images on your local computer as +`localhost:5000/yourname/container` and those images will be available inside +your kubernetes cluster with the same name. diff --git a/kubespray/docs/kubernetes-reliability.md b/kubespray/docs/kubernetes-reliability.md new file mode 100644 index 0000000..d4cca2e --- /dev/null +++ b/kubespray/docs/kubernetes-reliability.md @@ -0,0 +1,108 @@ +# Overview + +Distributed system such as Kubernetes are designed to be resilient to the +failures. More details about Kubernetes High-Availability (HA) may be found at +[Building High-Availability Clusters](https://kubernetes.io/docs/setup/production-environment/tools/kubeadm/high-availability/) + +To have a simple view the most of the parts of HA will be skipped to describe +Kubelet<->Controller Manager communication only. + +By default the normal behavior looks like: + +1. Kubelet updates it status to apiserver periodically, as specified by + `--node-status-update-frequency`. The default value is **10s**. + +2. Kubernetes controller manager checks the statuses of Kubelet every + `–-node-monitor-period`. The default value is **5s**. + +3. In case the status is updated within `--node-monitor-grace-period` of time, + Kubernetes controller manager considers healthy status of Kubelet. The + default value is **40s**. + +> Kubernetes controller manager and Kubelet work asynchronously. It means that +> the delay may include any network latency, API Server latency, etcd latency, +> latency caused by load on one's control plane nodes and so on. So if +> `--node-status-update-frequency` is set to 5s in reality it may appear in +> etcd in 6-7 seconds or even longer when etcd cannot commit data to quorum +> nodes. + +## Failure + +Kubelet will try to make `nodeStatusUpdateRetry` post attempts. Currently +`nodeStatusUpdateRetry` is constantly set to 5 in +[kubelet.go](https://github.com/kubernetes/kubernetes/blob/release-1.5/pkg/kubelet/kubelet.go#L102). + +Kubelet will try to update the status in +[tryUpdateNodeStatus](https://github.com/kubernetes/kubernetes/blob/release-1.5/pkg/kubelet/kubelet_node_status.go#L312) +function. Kubelet uses `http.Client()` Golang method, but has no specified +timeout. Thus there may be some glitches when API Server is overloaded while +TCP connection is established. + +So, there will be `nodeStatusUpdateRetry` * `--node-status-update-frequency` +attempts to set a status of node. + +At the same time Kubernetes controller manager will try to check +`nodeStatusUpdateRetry` times every `--node-monitor-period` of time. After +`--node-monitor-grace-period` it will consider node unhealthy. Pods will then be rescheduled based on the +[Taint Based Eviction](https://kubernetes.io/docs/concepts/scheduling-eviction/taint-and-toleration/#taint-based-evictions) +timers that you set on them individually, or the API Server's global timers:`--default-not-ready-toleration-seconds` & +``--default-unreachable-toleration-seconds``. + +Kube proxy has a watcher over API. Once pods are evicted, Kube proxy will +notice and will update iptables of the node. It will remove endpoints from +services so pods from failed node won't be accessible anymore. + +## Recommendations for different cases + +## Fast Update and Fast Reaction + +If `--node-status-update-frequency` is set to **4s** (10s is default). +`--node-monitor-period` to **2s** (5s is default). +`--node-monitor-grace-period` to **20s** (40s is default). +`--default-not-ready-toleration-seconds` and ``--default-unreachable-toleration-seconds`` are set to **30** +(300 seconds is default). Note these two values should be integers representing the number of seconds ("s" or "m" for +seconds\minutes are not specified). + +In such scenario, pods will be evicted in **50s** because the node will be +considered as down after **20s**, and `--default-not-ready-toleration-seconds` or +``--default-unreachable-toleration-seconds`` occur after **30s** more. However, this scenario creates an overhead on +etcd as every node will try to update its status every 2 seconds. + +If the environment has 1000 nodes, there will be 15000 node updates per +minute which may require large etcd containers or even dedicated nodes for etcd. + +> If we calculate the number of tries, the division will give 5, but in reality +> it will be from 3 to 5 with `nodeStatusUpdateRetry` attempts of each try. The +> total number of attempts will vary from 15 to 25 due to latency of all +> components. + +## Medium Update and Average Reaction + +Let's set `--node-status-update-frequency` to **20s** +`--node-monitor-grace-period` to **2m** and `--default-not-ready-toleration-seconds` and +``--default-unreachable-toleration-seconds`` to **60**. +In that case, Kubelet will try to update status every 20s. So, it will be 6 * 5 += 30 attempts before Kubernetes controller manager will consider unhealthy +status of node. After 1m it will evict all pods. The total time will be 3m +before eviction process. + +Such scenario is good for medium environments as 1000 nodes will require 3000 +etcd updates per minute. + +> In reality, there will be from 4 to 6 node update tries. The total number of +> of attempts will vary from 20 to 30. + +## Low Update and Slow reaction + +Let's set `--node-status-update-frequency` to **1m**. +`--node-monitor-grace-period` will set to **5m** and `--default-not-ready-toleration-seconds` and +``--default-unreachable-toleration-seconds`` to **60**. In this scenario, every kubelet will try to update the status +every minute. There will be 5 * 5 = 25 attempts before unhealthy status. After 5m, +Kubernetes controller manager will set unhealthy status. This means that pods +will be evicted after 1m after being marked unhealthy. (6m in total). + +> In reality, there will be from 3 to 5 tries. The total number of attempt will +> vary from 15 to 25. + +There can be different combinations such as Fast Update with Slow reaction to +satisfy specific cases. diff --git a/kubespray/docs/kylinlinux.md b/kubespray/docs/kylinlinux.md new file mode 100644 index 0000000..87dceff --- /dev/null +++ b/kubespray/docs/kylinlinux.md @@ -0,0 +1,11 @@ +# Kylin Linux + +Kylin Linux is supported with docker and containerd runtimes. + +**Note:** that Kylin Linux is not currently covered in kubespray CI and +support for it is currently considered experimental. + +At present, only `Kylin Linux Advanced Server V10 (Sword)` has been adapted, which can support the deployment of aarch64 and x86_64 platforms. + +There are no special considerations for using Kylin Linux as the target OS +for Kubespray deployments. diff --git a/kubespray/docs/large-deployments.md b/kubespray/docs/large-deployments.md new file mode 100644 index 0000000..7acbff3 --- /dev/null +++ b/kubespray/docs/large-deployments.md @@ -0,0 +1,52 @@ +Large deployments of K8s +======================== + +For a large scaled deployments, consider the following configuration changes: + +* Tune [ansible settings](https://docs.ansible.com/ansible/latest/intro_configuration.html) + for `forks` and `timeout` vars to fit large numbers of nodes being deployed. + +* Override containers' `foo_image_repo` vars to point to intranet registry. + +* Override the ``download_run_once: true`` and/or ``download_localhost: true``. + See download modes for details. + +* Adjust the `retry_stagger` global var as appropriate. It should provide sane + load on a delegate (the first K8s control plane node) then retrying failed + push or download operations. + +* Tune parameters for DNS related applications + Those are ``dns_replicas``, ``dns_cpu_limit``, + ``dns_cpu_requests``, ``dns_memory_limit``, ``dns_memory_requests``. + Please note that limits must always be greater than or equal to requests. + +* Tune CPU/memory limits and requests. Those are located in roles' defaults + and named like ``foo_memory_limit``, ``foo_memory_requests`` and + ``foo_cpu_limit``, ``foo_cpu_requests``. Note that 'Mi' memory units for K8s + will be submitted as 'M', if applied for ``docker run``, and cpu K8s units + will end up with the 'm' skipped for docker as well. This is required as + docker does not understand k8s units well. + +* Tune ``kubelet_status_update_frequency`` to increase reliability of kubelet. + ``kube_controller_node_monitor_grace_period``, + ``kube_controller_node_monitor_period``, + ``kube_apiserver_pod_eviction_not_ready_timeout_seconds`` & + ``kube_apiserver_pod_eviction_unreachable_timeout_seconds`` for better Kubernetes reliability. + Check out [Kubernetes Reliability](/docs/kubernetes-reliability.md) + +* Tune network prefix sizes. Those are ``kube_network_node_prefix``, + ``kube_service_addresses`` and ``kube_pods_subnet``. + +* Add calico_rr nodes if you are deploying with Calico or Canal. Nodes recover + from host/network interruption much quicker with calico_rr. + +* Check out the + [Inventory](/docs/getting-started.md#building-your-own-inventory) + section of the Getting started guide for tips on creating a large scale + Ansible inventory. + +* Override the ``etcd_events_cluster_setup: true`` store events in a separate + dedicated etcd instance. + +For example, when deploying 200 nodes, you may want to run ansible with +``--forks=50``, ``--timeout=600`` and define the ``retry_stagger: 60``. diff --git a/kubespray/docs/macvlan.md b/kubespray/docs/macvlan.md new file mode 100644 index 0000000..2d0de07 --- /dev/null +++ b/kubespray/docs/macvlan.md @@ -0,0 +1,41 @@ +# Macvlan + +## How to use it + +* Enable macvlan in `group_vars/k8s_cluster/k8s_cluster.yml` + +```yml +... +kube_network_plugin: macvlan +... +``` + +* Adjust the `macvlan_interface` in `group_vars/k8s_cluster/k8s-net-macvlan.yml` or by host in the `host.yml` file: + +```yml +all: + hosts: + node1: + ip: 10.2.2.1 + access_ip: 10.2.2.1 + ansible_host: 10.2.2.1 + macvlan_interface: ens5 +``` + +## Issue encountered + +* Service DNS + +reply from unexpected source: + +add `kube_proxy_masquerade_all: true` in `group_vars/all/all.yml` + +* Disable nodelocaldns + +The nodelocal dns IP is not reacheable. + +Disable it in `sample/group_vars/k8s_cluster/k8s_cluster.yml` + +```yml +enable_nodelocaldns: false +``` diff --git a/kubespray/docs/metallb.md b/kubespray/docs/metallb.md new file mode 100644 index 0000000..94b81fa --- /dev/null +++ b/kubespray/docs/metallb.md @@ -0,0 +1,222 @@ +# MetalLB + +MetalLB hooks into your Kubernetes cluster, and provides a network load-balancer implementation. +It allows you to create Kubernetes services of type "LoadBalancer" in clusters that don't run on a cloud provider, and thus cannot simply hook into 3rd party products to provide load-balancers. +The default operating mode of MetalLB is in ["Layer2"](https://metallb.universe.tf/concepts/layer2/) but it can also operate in ["BGP"](https://metallb.universe.tf/concepts/bgp/) mode. + +## Prerequisites + +You have to configure arp_ignore and arp_announce to avoid answering ARP queries from kube-ipvs0 interface for MetalLB to work. + +```yaml +kube_proxy_strict_arp: true +``` + +## Install + +You have to explicitly enable the MetalLB extension. + +```yaml +metallb_enabled: true +metallb_speaker_enabled: true +``` + +By default only the MetalLB BGP speaker is allowed to run on control plane nodes. If you have a single node cluster or a cluster where control plane are also worker nodes you may need to enable tolerations for the MetalLB controller: + +```yaml +metallb_config: + controller: + nodeselector: + kubernetes.io/os: linux + tolerations: + - key: "node-role.kubernetes.io/master" + operator: "Equal" + value: "" + effect: "NoSchedule" + - key: "node-role.kubernetes.io/control-plane" + operator: "Equal" + value: "" + effect: "NoSchedule" +``` + +If you'd like to set additional nodeSelector and tolerations values, you can do so in the following fasion: + +```yaml +metallb_config: + controller: + nodeselector: + kubernetes.io/os: linux + tolerations: + - key: "node-role.kubernetes.io/control-plane" + operator: "Equal" + value: "" + effect: "NoSchedule" + speaker: + nodeselector: + kubernetes.io/os: linux + tolerations: + - key: "node-role.kubernetes.io/control-plane" + operator: "Equal" + value: "" + effect: "NoSchedule" +``` + +## Pools + +First you need to specify all of the pools you are going to use: + +```yaml +metallb_config: + + address_pools: + + primary: + ip_range: + - 192.0.1.0-192.0.1.254 + + pool1: + ip_range: + - 192.0.2.1-192.0.2.1 + auto_assign: false # When set to false, you need to explicitly set the loadBalancerIP in the service! + + pool2: + ip_range: + - 192.0.3.0/24 + avoid_buggy_ips: true # When set to true, .0 and .255 addresses will be avoided. +``` + +## Layer2 Mode + +Pools that need to be configured in layer2 mode, need to be specified in a list: + +```yaml +metallb_config: + + layer2: + - primary +``` + +## BGP Mode + +When operating in BGP Mode MetalLB needs to have defined upstream peers and link the pool(s) specified above to the correct peer: + +```yaml +metallb_config: + + layer3: + defaults: + + peer_port: 179 # The TCP port to talk to. Defaults to 179, you shouldn't need to set this in production. + hold_time: 120s # Requested BGP hold time, per RFC4271. + + communities: + vpn-only: "1234:1" + NO_ADVERTISE: "65535:65282" + + metallb_peers: + + peer1: + peer_address: 192.0.2.1 + peer_asn: 64512 + my_asn: 4200000000 + communities: + - vpn-only + address_pool: + - pool1 + + # (optional) The source IP address to use when establishing the BGP session. In most cases the source-address field should only be used with per-node peers, i.e. peers with node selectors which select only one node. CURRENTLY NOT SUPPORTED + source_address: 192.0.2.2 + + # (optional) The router ID to use when connecting to this peer. Defaults to the node IP address. + # Generally only useful when you need to peer with another BGP router running on the same machine as MetalLB. + router_id: 1.2.3.4 + + # (optional) Password for TCPMD5 authenticated BGP sessions offered by some peers. + password: "changeme" + + peer2: + peer_address: 192.0.2.2 + peer_asn: 64513 + my_asn: 4200000000 + communities: + - NO_ADVERTISE + address_pool: + - pool2 + + # (optional) The source IP address to use when establishing the BGP session. In most cases the source-address field should only be used with per-node peers, i.e. peers with node selectors which select only one node. CURRENTLY NOT SUPPORTED + source_address: 192.0.2.1 + + # (optional) The router ID to use when connecting to this peer. Defaults to the node IP address. + # Generally only useful when you need to peer with another BGP router running on the same machine as MetalLB. + router_id: 1.2.3.5 + + # (optional) Password for TCPMD5 authenticated BGP sessions offered by some peers. + password: "changeme" +``` + +When using calico >= 3.18 you can replace MetalLB speaker by calico Service LoadBalancer IP advertisement. +See [calico service IPs advertisement documentation](https://docs.projectcalico.org/archive/v3.18/networking/advertise-service-ips#advertise-service-load-balancer-ip-addresses). +In this scenario you should disable the MetalLB speaker and configure the `calico_advertise_service_loadbalancer_ips` to match your `ip_range` + +```yaml +metallb_speaker_enabled: false +metallb_config: + address_pools: + primary: + ip_range: + - 10.5.0.0/16 + auto_assign: true + layer2: + - primary +calico_advertise_service_loadbalancer_ips: "{{ metallb_config.address_pools.primary.ip_range }}" +``` + +If you have additional loadbalancer IP pool in `metallb_config.address_pools` , ensure to add them to the list. + +```yaml +metallb_speaker_enabled: false +metallb_config: + address_pools: + primary: + ip_range: + - 10.5.0.0/16 + auto_assign: true + pool1: + ip_range: + - 10.6.0.0/16 + auto_assign: true + pool2: + ip_range: + - 10.10.0.0/16 + auto_assign: true + layer2: + - primary + layer3: + defaults: + peer_port: 179 + hold_time: 120s + communities: + vpn-only: "1234:1" + NO_ADVERTISE: "65535:65282" + metallb_peers: + peer1: + peer_address: 10.6.0.1 + peer_asn: 64512 + my_asn: 4200000000 + communities: + - vpn-only + address_pool: + - pool1 + peer2: + peer_address: 10.10.0.1 + peer_asn: 64513 + my_asn: 4200000000 + communities: + - NO_ADVERTISE + address_pool: + - pool2 +calico_advertise_service_loadbalancer_ips: + - 10.5.0.0/16 + - 10.6.0.0/16 + - 10.10.0.0/16 +``` diff --git a/kubespray/docs/mirror.md b/kubespray/docs/mirror.md new file mode 100644 index 0000000..3138d20 --- /dev/null +++ b/kubespray/docs/mirror.md @@ -0,0 +1,66 @@ +# Public Download Mirror + +The public mirror is useful to make the public resources download quickly in some areas of the world. (such as China). + +## Configuring Kubespray to use a mirror site + +You can follow the [offline](offline-environment.md) to config the image/file download configuration to the public mirror site. If you want to download quickly in China, the configuration can be like: + +```shell +gcr_image_repo: "gcr.m.daocloud.io" +kube_image_repo: "k8s.m.daocloud.io" +docker_image_repo: "docker.m.daocloud.io" +quay_image_repo: "quay.m.daocloud.io" +github_image_repo: "ghcr.m.daocloud.io" + +files_repo: "https://files.m.daocloud.io" +``` + +Use mirror sites only if you trust the provider. The Kubespray team cannot verify their reliability or security. +You can replace the `m.daocloud.io` with any site you want. + +## Example Usage Full Steps + +You can follow the full steps to use the kubesray with mirror. for example: + +Install Ansible according to Ansible installation guide then run the following steps: + +```shell +# Copy ``inventory/sample`` as ``inventory/mycluster`` +cp -rfp inventory/sample inventory/mycluster + +# Update Ansible inventory file with inventory builder +declare -a IPS=(10.10.1.3 10.10.1.4 10.10.1.5) +CONFIG_FILE=inventory/mycluster/hosts.yaml python3 contrib/inventory_builder/inventory.py ${IPS[@]} + +# Use the download mirror +cp inventory/mycluster/group_vars/all/offline.yml inventory/mycluster/group_vars/all/mirror.yml +sed -i -E '/# .*\{\{ files_repo/s/^# //g' inventory/mycluster/group_vars/all/mirror.yml +tee -a inventory/mycluster/group_vars/all/mirror.yml < diff --git a/kubespray/docs/netcheck.md b/kubespray/docs/netcheck.md new file mode 100644 index 0000000..6a1bf80 --- /dev/null +++ b/kubespray/docs/netcheck.md @@ -0,0 +1,41 @@ +# Network Checker Application + +With the ``deploy_netchecker`` var enabled (defaults to false), Kubespray deploys a +Network Checker Application from the 3rd side `mirantis/k8s-netchecker` docker +images. It consists of the server and agents trying to reach the server by usual +for Kubernetes applications network connectivity meanings. Therefore, this +automatically verifies a pod to pod connectivity via the cluster IP and checks +if DNS resolve is functioning as well. + +The checks are run by agents on a periodic basis and cover standard and host network +pods as well. The history of performed checks may be found in the agents' application +logs. + +To get the most recent and cluster-wide network connectivity report, run from +any of the cluster nodes: + +```ShellSession +curl http://localhost:31081/api/v1/connectivity_check +``` + +Note that Kubespray does not invoke the check but only deploys the application, if +requested. + +There are related application specific variables: + +```yml +netchecker_port: 31081 +agent_report_interval: 15 +netcheck_namespace: default +``` + +Note that the application verifies DNS resolve for FQDNs comprising only the +combination of the ``netcheck_namespace.dns_domain`` vars, for example the +``netchecker-service.default.svc.cluster.local``. If you want to deploy the application +to the non default namespace, make sure as well to adjust the ``searchdomains`` var +so the resulting search domain records to contain that namespace, like: + +```yml +search: foospace.cluster.local default.cluster.local ... +nameserver: ... +``` diff --git a/kubespray/docs/nodes.md b/kubespray/docs/nodes.md new file mode 100644 index 0000000..b9cef8d --- /dev/null +++ b/kubespray/docs/nodes.md @@ -0,0 +1,185 @@ +# Adding/replacing a node + +Modified from [comments in #3471](https://github.com/kubernetes-sigs/kubespray/issues/3471#issuecomment-530036084) + +## Limitation: Removal of first kube_control_plane and etcd-master + +Currently you can't remove the first node in your kube_control_plane and etcd-master list. If you still want to remove this node you have to: + +### 1) Change order of current control planes + +Modify the order of your control plane list by pushing your first entry to any other position. E.g. if you want to remove `node-1` of the following example: + +```yaml + children: + kube_control_plane: + hosts: + node-1: + node-2: + node-3: + kube_node: + hosts: + node-1: + node-2: + node-3: + etcd: + hosts: + node-1: + node-2: + node-3: +``` + +change your inventory to: + +```yaml + children: + kube_control_plane: + hosts: + node-2: + node-3: + node-1: + kube_node: + hosts: + node-2: + node-3: + node-1: + etcd: + hosts: + node-2: + node-3: + node-1: +``` + +## 2) Upgrade the cluster + +run `upgrade-cluster.yml` or `cluster.yml`. Now you are good to go on with the removal. + +## Adding/replacing a worker node + +This should be the easiest. + +### 1) Add new node to the inventory + +### 2) Run `scale.yml` + +You can use `--limit=NODE_NAME` to limit Kubespray to avoid disturbing other nodes in the cluster. + +Before using `--limit` run playbook `facts.yml` without the limit to refresh facts cache for all nodes. + +### 3) Remove an old node with remove-node.yml + +With the old node still in the inventory, run `remove-node.yml`. You need to pass `-e node=NODE_NAME` to the playbook to limit the execution to the node being removed. + +If the node you want to remove is not online, you should add `reset_nodes=false` and `allow_ungraceful_removal=true` to your extra-vars: `-e node=NODE_NAME -e reset_nodes=false -e allow_ungraceful_removal=true`. +Use this flag even when you remove other types of nodes like a control plane or etcd nodes. + +### 4) Remove the node from the inventory + +That's it. + +## Adding/replacing a control plane node + +### 1) Run `cluster.yml` + +Append the new host to the inventory and run `cluster.yml`. You can NOT use `scale.yml` for that. + +### 2) Restart kube-system/nginx-proxy + +In all hosts, restart nginx-proxy pod. This pod is a local proxy for the apiserver. Kubespray will update its static config, but it needs to be restarted in order to reload. + +```sh +# run in every host +docker ps | grep k8s_nginx-proxy_nginx-proxy | awk '{print $1}' | xargs docker restart + +# or with containerd +crictl ps | grep nginx-proxy | awk '{print $1}' | xargs crictl stop +``` + +### 3) Remove old control plane nodes + +With the old node still in the inventory, run `remove-node.yml`. You need to pass `-e node=NODE_NAME` to the playbook to limit the execution to the node being removed. +If the node you want to remove is not online, you should add `reset_nodes=false` and `allow_ungraceful_removal=true` to your extra-vars. + +## Replacing a first control plane node + +### 1) Change control plane nodes order in inventory + +from + +```ini +[kube_control_plane] + node-1 + node-2 + node-3 +``` + +to + +```ini +[kube_control_plane] + node-2 + node-3 + node-1 +``` + +### 2) Remove old first control plane node from cluster + +With the old node still in the inventory, run `remove-node.yml`. You need to pass `-e node=node-1` to the playbook to limit the execution to the node being removed. +If the node you want to remove is not online, you should add `reset_nodes=false` and `allow_ungraceful_removal=true` to your extra-vars. + +### 3) Edit cluster-info configmap in kube-public namespace + +`kubectl edit cm -n kube-public cluster-info` + +Change ip of old kube_control_plane node with ip of live kube_control_plane node (`server` field). Also, update `certificate-authority-data` field if you changed certs. + +### 4) Add new control plane node + +Update inventory (if needed) + +Run `cluster.yml` with `--limit=kube_control_plane` + +## Adding an etcd node + +You need to make sure there are always an odd number of etcd nodes in the cluster. In such a way, this is always a replacement or scale up operation. Either add two new nodes or remove an old one. + +### 1) Add the new node running cluster.yml + +Update the inventory and run `cluster.yml` passing `--limit=etcd,kube_control_plane -e ignore_assert_errors=yes`. +If the node you want to add as an etcd node is already a worker or control plane node in your cluster, you have to remove him first using `remove-node.yml`. + +Run `upgrade-cluster.yml` also passing `--limit=etcd,kube_control_plane -e ignore_assert_errors=yes`. This is necessary to update all etcd configuration in the cluster. + +At this point, you will have an even number of nodes. +Everything should still be working, and you should only have problems if the cluster decides to elect a new etcd leader before you remove a node. +Even so, running applications should continue to be available. + +If you add multiple etcd nodes with one run, you might want to append `-e etcd_retries=10` to increase the amount of retries between each etcd node join. +Otherwise the etcd cluster might still be processing the first join and fail on subsequent nodes. `etcd_retries=10` might work to join 3 new nodes. + +### 2) Add the new node to apiserver config + +In every control plane node, edit `/etc/kubernetes/manifests/kube-apiserver.yaml`. Make sure the new etcd nodes are present in the apiserver command line parameter `--etcd-servers=...`. + +## Removing an etcd node + +### 1) Remove an old etcd node + +With the node still in the inventory, run `remove-node.yml` passing `-e node=NODE_NAME` as the name of the node that should be removed. +If the node you want to remove is not online, you should add `reset_nodes=false` and `allow_ungraceful_removal=true` to your extra-vars. + +### 2) Make sure only remaining nodes are in your inventory + +Remove `NODE_NAME` from your inventory file. + +### 3) Update kubernetes and network configuration files with the valid list of etcd members + +Run `cluster.yml` to regenerate the configuration files on all remaining nodes. + +### 4) Remove the old etcd node from apiserver config + +In every control plane node, edit `/etc/kubernetes/manifests/kube-apiserver.yaml`. Make sure only active etcd nodes are still present in the apiserver command line parameter `--etcd-servers=...`. + +### 5) Shutdown the old instance + +That's it. diff --git a/kubespray/docs/ntp.md b/kubespray/docs/ntp.md new file mode 100644 index 0000000..a91e09e --- /dev/null +++ b/kubespray/docs/ntp.md @@ -0,0 +1,50 @@ +# NTP synchronization + +The Network Time Protocol (NTP) is a networking protocol for clock synchronization between computer systems. Time synchronization is important to Kubernetes and Etcd. + +## Enable the NTP + +To start the ntpd(or chrony) service and enable it at system boot. There are related specific variables: + +```ShellSession +ntp_enabled: true +``` + +The NTP service would be enabled and sync time automatically. + +## Customize the NTP configure file + +In the Air-Gap environment, the node cannot access the NTP server by internet. So the node can use the customized ntp server by configuring ntp file. + +```ShellSession +ntp_enabled: true +ntp_manage_config: true +ntp_servers: + - "0.your-ntp-server.org iburst" + - "1.your-ntp-server.org iburst" + - "2.your-ntp-server.org iburst" + - "3.your-ntp-server.org iburst" +``` + +## Setting the TimeZone + +The timezone can also be set by the `ntp_timezone` , eg: "Etc/UTC","Asia/Shanghai". If not set, the timezone will not change. + +```ShellSession +ntp_enabled: true +ntp_timezone: Etc/UTC +``` + +## Advanced Configure + +Enable `tinker panic` is useful when running NTP in a VM environment to avoiding clock drift on VMs. It only takes effect when ntp_manage_config is true. + +```ShellSession +ntp_tinker_panic: true +``` + +Force sync time immediately by NTP after the ntp installed, which is useful in newly installed system. + +```ShellSession +ntp_force_sync_immediately: true +``` diff --git a/kubespray/docs/offline-environment.md b/kubespray/docs/offline-environment.md new file mode 100644 index 0000000..743c2f7 --- /dev/null +++ b/kubespray/docs/offline-environment.md @@ -0,0 +1,152 @@ +# Offline environment + +In case your servers don't have access to the internet directly (for example +when deploying on premises with security constraints), you need to get the +following artifacts in advance from another environment where has access to the internet. + +* Some static files (zips and binaries) +* OS packages (rpm/deb files) +* Container images used by Kubespray. Exhaustive list depends on your setup +* [Optional] Python packages used by Kubespray (only required if your OS doesn't provide all python packages/versions + listed in `requirements.txt`) +* [Optional] Helm chart files (only required if `helm_enabled=true`) + +Then you need to setup the following services on your offline environment: + +* an HTTP reverse proxy/cache/mirror to serve some static files (zips and binaries) +* an internal Yum/Deb repository for OS packages +* an internal container image registry that need to be populated with all container images used by Kubespray +* [Optional] an internal PyPi server for python packages used by Kubespray +* [Optional] an internal Helm registry for Helm chart files + +You can get artifact lists with [generate_list.sh](/contrib/offline/generate_list.sh) script. +In addition, you can find some tools for offline deployment under [contrib/offline](/contrib/offline/README.md). + +## Configure Inventory + +Once all artifacts are accessible from your internal network, **adjust** the following variables +in [your inventory](/inventory/sample/group_vars/all/offline.yml) to match your environment: + +```yaml +# Registry overrides +kube_image_repo: "{{ registry_host }}" +gcr_image_repo: "{{ registry_host }}" +docker_image_repo: "{{ registry_host }}" +quay_image_repo: "{{ registry_host }}" +github_image_repo: "{{ registry_host }}" + +kubeadm_download_url: "{{ files_repo }}/kubernetes/{{ kube_version }}/kubeadm" +kubectl_download_url: "{{ files_repo }}/kubernetes/{{ kube_version }}/kubectl" +kubelet_download_url: "{{ files_repo }}/kubernetes/{{ kube_version }}/kubelet" +# etcd is optional if you **DON'T** use etcd_deployment=host +etcd_download_url: "{{ files_repo }}/kubernetes/etcd/etcd-{{ etcd_version }}-linux-{{ image_arch }}.tar.gz" +cni_download_url: "{{ files_repo }}/kubernetes/cni/cni-plugins-linux-{{ image_arch }}-{{ cni_version }}.tgz" +crictl_download_url: "{{ files_repo }}/kubernetes/cri-tools/crictl-{{ crictl_version }}-{{ ansible_system | lower }}-{{ image_arch }}.tar.gz" +# If using Calico +calicoctl_download_url: "{{ files_repo }}/kubernetes/calico/{{ calico_ctl_version }}/calicoctl-linux-{{ image_arch }}" +# If using Calico with kdd +calico_crds_download_url: "{{ files_repo }}/kubernetes/calico/{{ calico_version }}.tar.gz" +# Containerd +containerd_download_url: "{{ files_repo }}/containerd-{{ containerd_version }}-linux-{{ image_arch }}.tar.gz" +runc_download_url: "{{ files_repo }}/runc.{{ image_arch }}" +nerdctl_download_url: "{{ files_repo }}/nerdctl-{{ nerdctl_version }}-{{ ansible_system | lower }}-{{ image_arch }}.tar.gz" +# Insecure registries for containerd +containerd_registries_mirrors: + - prefix: "{{ registry_addr }}" + mirrors: + - host: "{{ registry_host }}" + capabilities: ["pull", "resolve"] + skip_verify: true + +# CentOS/Redhat/AlmaLinux/Rocky Linux +## Docker / Containerd +docker_rh_repo_base_url: "{{ yum_repo }}/docker-ce/$releasever/$basearch" +docker_rh_repo_gpgkey: "{{ yum_repo }}/docker-ce/gpg" + +# Fedora +## Docker +docker_fedora_repo_base_url: "{{ yum_repo }}/docker-ce/{{ ansible_distribution_major_version }}/{{ ansible_architecture }}" +docker_fedora_repo_gpgkey: "{{ yum_repo }}/docker-ce/gpg" +## Containerd +containerd_fedora_repo_base_url: "{{ yum_repo }}/containerd" +containerd_fedora_repo_gpgkey: "{{ yum_repo }}/docker-ce/gpg" + +# Debian +## Docker +docker_debian_repo_base_url: "{{ debian_repo }}/docker-ce" +docker_debian_repo_gpgkey: "{{ debian_repo }}/docker-ce/gpg" +## Containerd +containerd_debian_repo_base_url: "{{ ubuntu_repo }}/containerd" +containerd_debian_repo_gpgkey: "{{ ubuntu_repo }}/containerd/gpg" +containerd_debian_repo_repokey: 'YOURREPOKEY' + +# Ubuntu +## Docker +docker_ubuntu_repo_base_url: "{{ ubuntu_repo }}/docker-ce" +docker_ubuntu_repo_gpgkey: "{{ ubuntu_repo }}/docker-ce/gpg" +## Containerd +containerd_ubuntu_repo_base_url: "{{ ubuntu_repo }}/containerd" +containerd_ubuntu_repo_gpgkey: "{{ ubuntu_repo }}/containerd/gpg" +containerd_ubuntu_repo_repokey: 'YOURREPOKEY' +``` + +For the OS specific settings, just define the one matching your OS. +If you use the settings like the one above, you'll need to define in your inventory the following variables: + +* `registry_host`: Container image registry. If you _don't_ use the same repository path for the container images that + the ones defined + in [Download's role defaults](https://github.com/kubernetes-sigs/kubespray/blob/master/roles/download/defaults/main/main.yml) + , you need to override the `*_image_repo` for these container images. If you want to make your life easier, use the + same repository path, you won't have to override anything else. +* `registry_addr`: Container image registry, but only have [domain or ip]:[port]. +* `files_repo`: HTTP webserver or reverse proxy that is able to serve the files listed above. Path is not important, you + can store them anywhere as long as it's accessible by kubespray. It's recommended to use `*_version` in the path so + that you don't need to modify this setting everytime kubespray upgrades one of these components. +* `yum_repo`/`debian_repo`/`ubuntu_repo`: OS package repository depending on your OS, should point to your internal + repository. Adjust the path accordingly. + +## Install Kubespray Python Packages + +### Recommended way: Kubespray Container Image + +The easiest way is to use [kubespray container image](https://quay.io/kubespray/kubespray) as all the required packages +are baked in the image. +Just copy the container image in your private container image registry and you are all set! + +### Manual installation + +Look at the `requirements.txt` file and check if your OS provides all packages out-of-the-box (Using the OS package +manager). For those missing, you need to either use a proxy that has Internet access (typically from a DMZ) or setup a +PyPi server in your network that will host these packages. + +If you're using an HTTP(S) proxy to download your python packages: + +```bash +sudo pip install --proxy=https://[username:password@]proxyserver:port -r requirements.txt +``` + +When using an internal PyPi server: + +```bash +# If you host all required packages +pip install -i https://pypiserver/pypi -r requirements.txt + +# If you only need the ones missing from the OS package manager +pip install -i https://pypiserver/pypi package_you_miss +``` + +## Run Kubespray as usual + +Once all artifacts are in place and your inventory properly set up, you can run kubespray with the +regular `cluster.yaml` command: + +```bash +ansible-playbook -i inventory/my_airgap_cluster/hosts.yaml -b cluster.yml +``` + +If you use [Kubespray Container Image](#recommended-way:-kubespray-container-image), you can mount your inventory inside +the container: + +```bash +docker run --rm -it -v path_to_inventory/my_airgap_cluster:inventory/my_airgap_cluster myprivateregisry.com/kubespray/kubespray:v2.14.0 ansible-playbook -i inventory/my_airgap_cluster/hosts.yaml -b cluster.yml +``` diff --git a/kubespray/docs/openeuler.md b/kubespray/docs/openeuler.md new file mode 100644 index 0000000..c585d37 --- /dev/null +++ b/kubespray/docs/openeuler.md @@ -0,0 +1,11 @@ +# OpenEuler + +[OpenEuler](https://www.openeuler.org/en/) Linux is supported with docker and containerd runtimes. + +**Note:** that OpenEuler Linux is not currently covered in kubespray CI and +support for it is currently considered experimental. + +At present, only `openEuler 22.03 LTS` has been adapted, which can support the deployment of aarch64 and x86_64 platforms. + +There are no special considerations for using OpenEuler Linux as the target OS +for Kubespray deployments. diff --git a/kubespray/docs/openstack.md b/kubespray/docs/openstack.md new file mode 100644 index 0000000..0e37a6c --- /dev/null +++ b/kubespray/docs/openstack.md @@ -0,0 +1,158 @@ + +# OpenStack + +## Known compatible public clouds + +Kubespray has been tested on a number of OpenStack Public Clouds including (in alphabetical order): + +- [Auro](https://auro.io/) +- [Betacloud](https://www.betacloud.io/) +- [CityCloud](https://www.citycloud.com/) +- [DreamHost](https://www.dreamhost.com/cloud/computing/) +- [ELASTX](https://elastx.se/) +- [EnterCloudSuite](https://www.entercloudsuite.com/) +- [FugaCloud](https://fuga.cloud/) +- [Infomaniak](https://infomaniak.com) +- [Open Telekom Cloud](https://cloud.telekom.de/) : requires to set the variable `wait_for_floatingip = "true"` in your cluster.tfvars +- [OVHcloud](https://www.ovhcloud.com/) +- [Rackspace](https://www.rackspace.com/) +- [Ultimum](https://ultimum.io/) +- [VexxHost](https://vexxhost.com/) +- [Zetta](https://www.zetta.io/) + +## The in-tree cloud provider + +To deploy Kubespray on [OpenStack](https://www.openstack.org/) uncomment the `cloud_provider` option in `group_vars/all/all.yml` and set it to `openstack`. + +After that make sure to source in your OpenStack credentials like you would do when using `nova-client` or `neutron-client` by using `source path/to/your/openstack-rc` or `. path/to/your/openstack-rc`. + +For those who prefer to pass the OpenStack CA certificate as a string, one can +base64 encode the cacert file and store it in the variable `openstack_cacert`. + +The next step is to make sure the hostnames in your `inventory` file are identical to your instance names in OpenStack. +Otherwise [cinder](https://wiki.openstack.org/wiki/Cinder) won't work as expected. + +Unless you are using calico or kube-router you can now run the playbook. + +## The external cloud provider + +The in-tree cloud provider is deprecated and will be removed in a future version of Kubernetes. The target release for removing all remaining in-tree cloud providers is set to 1.21. + +The new cloud provider is configured to have Octavia by default in Kubespray. + +- Enable the new external cloud provider in `group_vars/all/all.yml`: + + ```yaml + cloud_provider: external + external_cloud_provider: openstack + ``` + +- Enable Cinder CSI in `group_vars/all/openstack.yml`: + + ```yaml + cinder_csi_enabled: true + ``` + +- Enable topology support (optional), if your openstack provider has custom Zone names you can override the default "nova" zone by setting the variable `cinder_topology_zones` + + ```yaml + cinder_topology: true + ``` + +- Enabling `cinder_csi_ignore_volume_az: true`, ignores volumeAZ and schedules on any of the available node AZ. + + ```yaml + cinder_csi_ignore_volume_az: true + ``` + +- If you are using OpenStack loadbalancer(s) replace the `openstack_lbaas_subnet_id` with the new `external_openstack_lbaas_subnet_id`. **Note** The new cloud provider is using Octavia instead of Neutron LBaaS by default! +- Enable 3 feature gates to allow migration of all volumes and storage classes (if you have any feature gates already set just add the 3 listed below): + + ```yaml + kube_feature_gates: + - CSIMigration=true + - CSIMigrationOpenStack=true + - ExpandCSIVolumes=true + ``` + +- If you are in a case of a multi-nic OpenStack VMs (see [kubernetes/cloud-provider-openstack#407](https://github.com/kubernetes/cloud-provider-openstack/issues/407) and [#6083](https://github.com/kubernetes-sigs/kubespray/issues/6083) for explanation), you should override the default OpenStack networking configuration: + + ```yaml + external_openstack_network_ipv6_disabled: false + external_openstack_network_internal_networks: [] + external_openstack_network_public_networks: [] + ``` + +- You can override the default OpenStack metadata configuration (see [#6338](https://github.com/kubernetes-sigs/kubespray/issues/6338) for explanation): + + ```yaml + external_openstack_metadata_search_order: "configDrive,metadataService" + ``` + +- Available variables for configuring lbaas: + + ```yaml + external_openstack_lbaas_enabled: true + external_openstack_lbaas_floating_network_id: "Neutron network ID to get floating IP from" + external_openstack_lbaas_floating_subnet_id: "Neutron subnet ID to get floating IP from" + external_openstack_lbaas_method: ROUND_ROBIN + external_openstack_lbaas_provider: amphora + external_openstack_lbaas_subnet_id: "Neutron subnet ID to create LBaaS VIP" + external_openstack_lbaas_network_id: "Neutron network ID to create LBaaS VIP" + external_openstack_lbaas_manage_security_groups: false + external_openstack_lbaas_create_monitor: false + external_openstack_lbaas_monitor_delay: 5 + external_openstack_lbaas_monitor_max_retries: 1 + external_openstack_lbaas_monitor_timeout: 3 + external_openstack_lbaas_internal_lb: false + + ``` + +- Run `source path/to/your/openstack-rc` to read your OpenStack credentials like `OS_AUTH_URL`, `OS_USERNAME`, `OS_PASSWORD`, etc. Those variables are used for accessing OpenStack from the external cloud provider. +- Run the `cluster.yml` playbook + +## Additional step needed when using calico or kube-router + +Being L3 CNI, calico and kube-router do not encapsulate all packages with the hosts' ip addresses. Instead the packets will be routed with the PODs ip addresses directly. + +OpenStack will filter and drop all packets from ips it does not know to prevent spoofing. + +In order to make L3 CNIs work on OpenStack you will need to tell OpenStack to allow pods packets by allowing the network they use. + +First you will need the ids of your OpenStack instances that will run kubernetes: + + ```bash + openstack server list --project YOUR_PROJECT + +--------------------------------------+--------+----------------------------------+--------+-------------+ + | ID | Name | Tenant ID | Status | Power State | + +--------------------------------------+--------+----------------------------------+--------+-------------+ + | e1f48aad-df96-4bce-bf61-62ae12bf3f95 | k8s-1 | fba478440cb2444a9e5cf03717eb5d6f | ACTIVE | Running | + | 725cd548-6ea3-426b-baaa-e7306d3c8052 | k8s-2 | fba478440cb2444a9e5cf03717eb5d6f | ACTIVE | Running | + ``` + +Then you can use the instance ids to find the connected [neutron](https://wiki.openstack.org/wiki/Neutron) ports (though they are now configured through using OpenStack): + + ```bash + openstack port list -c id -c device_id --project YOUR_PROJECT + +--------------------------------------+--------------------------------------+ + | id | device_id | + +--------------------------------------+--------------------------------------+ + | 5662a4e0-e646-47f0-bf88-d80fbd2d99ef | e1f48aad-df96-4bce-bf61-62ae12bf3f95 | + | e5ae2045-a1e1-4e99-9aac-4353889449a7 | 725cd548-6ea3-426b-baaa-e7306d3c8052 | + ``` + +Given the port ids on the left, you can set the two `allowed-address`(es) in OpenStack. Note that you have to allow both `kube_service_addresses` (default `10.233.0.0/18`) and `kube_pods_subnet` (default `10.233.64.0/18`.) + + ```bash + # allow kube_service_addresses and kube_pods_subnet network + openstack port set 5662a4e0-e646-47f0-bf88-d80fbd2d99ef --allowed-address ip-address=10.233.0.0/18 --allowed-address ip-address=10.233.64.0/18 + openstack port set e5ae2045-a1e1-4e99-9aac-4353889449a7 --allowed-address ip-address=10.233.0.0/18 --allowed-address ip-address=10.233.64.0/18 + ``` + +If all the VMs in the tenant correspond to Kubespray deployment, you can "sweep run" above with: + + ```bash + openstack port list --device-owner=compute:nova -c ID -f value | xargs -tI@ openstack port set @ --allowed-address ip-address=10.233.0.0/18 --allowed-address ip-address=10.233.64.0/18 + ``` + +Now you can finally run the playbook. diff --git a/kubespray/docs/opensuse.md b/kubespray/docs/opensuse.md new file mode 100644 index 0000000..47b01f0 --- /dev/null +++ b/kubespray/docs/opensuse.md @@ -0,0 +1,17 @@ +# openSUSE Leap 15.3 and Tumbleweed + +openSUSE Leap installation Notes: + +- Install Ansible + + ```ShellSession + sudo zypper ref + sudo zypper -n install ansible + + ``` + +- Install Jinja2 and Python-Netaddr + + ```sudo zypper -n install python-Jinja2 python-netaddr``` + +Now you can continue with [Preparing your deployment](getting-started.md#starting-custom-deployment) diff --git a/kubespray/docs/port-requirements.md b/kubespray/docs/port-requirements.md new file mode 100644 index 0000000..4a2f063 --- /dev/null +++ b/kubespray/docs/port-requirements.md @@ -0,0 +1,70 @@ +# Port Requirements + +To operate properly, Kubespray requires some ports to be opened. If the network is configured with firewall rules, it is needed to ensure infrastructure components can communicate with each other through specific ports. + +Ensure the following ports required by Kubespray are open on the network and configured to allow access between hosts. Some ports are optional depending on the configuration and usage. + +## Kubernetes + +### Control plane + +| Protocol | Port | Description | +|----------|--------| ------------ | +| TCP | 22 | ssh for ansible | +| TCP | 2379 | etcd client port| +| TCP | 2380 | etcd peer port | +| TCP | 6443 | kubernetes api | +| TCP | 10250 | kubelet api | +| TCP | 10257 | kube-scheduler | +| TCP | 10259 | kube-controller-manager | + +### Worker node(s) + +| Protocol | Port | Description | +|----------|-------- | ------------ | +| TCP | 22 | ssh for ansible | +| TCP | 10250 | kubelet api | +| TCP | 30000-32767| kube nodePort range | + +refers to: [Kubernetes Docs](https://kubernetes.io/docs/reference/networking/ports-and-protocols/) + +## Calico + +If Calico is used, it requires: + +| Protocol | Port | Description | +|----------|-------- | ------------ | +| TCP | 179 | Calico networking (BGP) | +| UDP | 4789 | Calico CNI with VXLAN enabled | +| TCP | 5473 | Calico CNI with Typha enabled | +| UDP | 51820 | Calico with IPv4 Wireguard enabled | +| UDP | 51821 | Calico with IPv6 Wireguard enabled | +| IPENCAP / IPIP | - | Calico CNI with IPIP enabled | + +refers to: [Calico Docs](https://docs.tigera.io/calico/latest/getting-started/kubernetes/requirements#network-requirements) + +## Cilium + +If Cilium is used, it requires: + +| Protocol | Port | Description | +|----------|-------- | ------------ | +| TCP | 4240 | Cilium Health checks (``cilium-health``) | +| TCP | 4244 | Hubble server | +| TCP | 4245 | Hubble Relay | +| UDP | 8472 | VXLAN overlay | +| TCP | 9962 | Cilium-agent Prometheus metrics | +| TCP | 9963 | Cilium-operator Prometheus metrics | +| TCP | 9964 | Cilium-proxy Prometheus metrics | +| UDP | 51871 | WireGuard encryption tunnel endpoint | +| ICMP | - | health checks | + +refers to: [Cilium Docs](https://docs.cilium.io/en/v1.13/operations/system_requirements/) + +## Addons + +| Protocol | Port | Description | +|----------|-------- | ------------ | +| TCP | 9100 | node exporter | +| TCP/UDP | 7472 | metallb metrics ports | +| TCP/UDP | 7946 | metallb L2 operating mode | diff --git a/kubespray/docs/proxy.md b/kubespray/docs/proxy.md new file mode 100644 index 0000000..aea84c1 --- /dev/null +++ b/kubespray/docs/proxy.md @@ -0,0 +1,29 @@ +# Setting up Environment Proxy + +If you set http and https proxy, all nodes and loadbalancer will be excluded from proxy with generating no_proxy variable in `roles/kubespray-defaults/tasks/no_proxy.yml`, if you have additional resources for exclude add them to `additional_no_proxy` variable. If you want fully override your `no_proxy` setting, then fill in just `no_proxy` and no nodes or loadbalancer addresses will be added to no_proxy. + +## Set proxy for http and https + + `http_proxy:"http://example.proxy.tld:port"` + `https_proxy:"http://example.proxy.tld:port"` + +## Set custom CA + +CA must be already on each target nodes + + `https_proxy_cert_file: /path/to/host/custom/ca.crt` + +## Set default no_proxy (this will override default no_proxy generation) + +`no_proxy: "node1,node1_ip,node2,node2_ip...additional_host"` + +## Set additional addresses to default no_proxy (all cluster nodes and loadbalancer) + +`additional_no_proxy: "additional_host1,additional_host2"` + +## Exclude workers from no_proxy + +Since workers are included in the no_proxy variable, by default, docker engine will be restarted on all nodes (all +pods will restart) when adding or removing workers. To override this behaviour by only including control plane nodes in the +no_proxy variable, set: +`no_proxy_exclude_workers: true` diff --git a/kubespray/docs/recover-control-plane.md b/kubespray/docs/recover-control-plane.md new file mode 100644 index 0000000..0b80da2 --- /dev/null +++ b/kubespray/docs/recover-control-plane.md @@ -0,0 +1,42 @@ + +# Recovering the control plane + +To recover from broken nodes in the control plane use the "recover\-control\-plane.yml" playbook. + +* Backup what you can +* Provision new nodes to replace the broken ones +* Place the surviving nodes of the control plane first in the "etcd" and "kube\_control\_plane" groups +* Add the new nodes below the surviving control plane nodes in the "etcd" and "kube\_control\_plane" groups + +Examples of what broken means in this context: + +* One or more bare metal node(s) suffer from unrecoverable hardware failure +* One or more node(s) fail during patching or upgrading +* Etcd database corruption +* Other node related failures leaving your control plane degraded or nonfunctional + +__Note that you need at least one functional node to be able to recover using this method.__ + +## Runbook + +* Move any broken etcd nodes into the "broken\_etcd" group, make sure the "etcd\_member\_name" variable is set. +* Move any broken control plane nodes into the "broken\_kube\_control\_plane" group. + +Then run the playbook with ```--limit etcd,kube_control_plane``` and increase the number of ETCD retries by setting ```-e etcd_retries=10``` or something even larger. The amount of retries required is difficult to predict. + +When finished you should have a fully working control plane again. + +## Recover from lost quorum + +The playbook attempts to figure out it the etcd quorum is intact. If quorum is lost it will attempt to take a snapshot from the first node in the "etcd" group and restore from that. If you would like to restore from an alternate snapshot set the path to that snapshot in the "etcd\_snapshot" variable. + +```-e etcd_snapshot=/tmp/etcd_snapshot``` + +## Caveats + +* The playbook has only been tested with fairly small etcd databases. +* If your new control plane nodes have new ip addresses you may have to change settings in various places. +* There may be disruptions while running the playbook. +* There are absolutely no guarantees. + +If possible try to break a cluster in the same way that your target cluster is broken and test to recover that before trying on the real target cluster. diff --git a/kubespray/docs/rhel.md b/kubespray/docs/rhel.md new file mode 100644 index 0000000..f8a827d --- /dev/null +++ b/kubespray/docs/rhel.md @@ -0,0 +1,34 @@ +# Red Hat Enterprise Linux (RHEL) + +## RHEL Support Subscription Registration + +In order to install packages via yum or dnf, RHEL 7/8 hosts are required to be registered for a valid Red Hat support subscription. + +You can apply for a 1-year Development support subscription by creating a [Red Hat Developers](https://developers.redhat.com/) account. Be aware though that as the Red Hat Developers subscription is limited to only 1 year, it should not be used to register RHEL 7/8 hosts provisioned in Production environments. + +Once you have a Red Hat support account, simply add the credentials to the Ansible inventory parameters `rh_subscription_username` and `rh_subscription_password` prior to deploying Kubespray. If your company has a Corporate Red Hat support account, then obtain an **Organization ID** and **Activation Key**, and add these to the Ansible inventory parameters `rh_subscription_org_id` and `rh_subscription_activation_key` instead of using your Red Hat support account credentials. + +```ini +rh_subscription_username: "" +rh_subscription_password: "" +# rh_subscription_org_id: "" +# rh_subscription_activation_key: "" +``` + +Either the Red Hat support account username/password, or Organization ID/Activation Key combination must be specified in the Ansible inventory in order for the Red Hat subscription registration to complete successfully during the deployment of Kubespray. + +Update the Ansible inventory parameters `rh_subscription_usage`, `rh_subscription_role` and `rh_subscription_sla` if necessary to suit your specific requirements. + +```ini +rh_subscription_usage: "Development" +rh_subscription_role: "Red Hat Enterprise Server" +rh_subscription_sla: "Self-Support" +``` + +If the RHEL 7/8 hosts are already registered to a valid Red Hat support subscription via an alternative configuration management approach prior to the deployment of Kubespray, the successful RHEL `subscription-manager` status check will simply result in the RHEL subscription registration tasks being skipped. + +## RHEL 8 + +If you have containers that are using iptables in the host network namespace (`hostNetwork=true`), +you need to ensure they are using iptables-nft. +An example how k8s do the autodetection can be found [in this PR](https://github.com/kubernetes/kubernetes/pull/82966) diff --git a/kubespray/docs/roadmap.md b/kubespray/docs/roadmap.md new file mode 100644 index 0000000..78166b8 --- /dev/null +++ b/kubespray/docs/roadmap.md @@ -0,0 +1,3 @@ +# Kubespray's roadmap + +We are tracking the evolution towards Kubespray 3.0 in [#6400](https://github.com/kubernetes-sigs/kubespray/issues/6400) as well as in other open issue in our [github issues](https://github.com/kubernetes-sigs/kubespray/issues/) section. diff --git a/kubespray/docs/setting-up-your-first-cluster.md b/kubespray/docs/setting-up-your-first-cluster.md new file mode 100644 index 0000000..a8200a3 --- /dev/null +++ b/kubespray/docs/setting-up-your-first-cluster.md @@ -0,0 +1,642 @@ +# Setting up your first cluster with Kubespray + +This tutorial walks you through the detailed steps for setting up Kubernetes +with [Kubespray](https://kubespray.io/). + +The guide is inspired on the tutorial [Kubernetes The Hard Way](https://github.com/kelseyhightower/kubernetes-the-hard-way), with the +difference that here we want to showcase how to spin up a Kubernetes cluster +in a more managed fashion with Kubespray. + +## Target Audience + +The target audience for this tutorial is someone looking for a +hands-on guide to get started with Kubespray. + +## Cluster Details + +* [kubespray](https://github.com/kubernetes-sigs/kubespray) +* [kubernetes](https://github.com/kubernetes/kubernetes) + +## Prerequisites + +* Google Cloud Platform: This tutorial leverages the [Google Cloud Platform](https://cloud.google.com/) to streamline provisioning of the compute infrastructure required to bootstrap a Kubernetes cluster from the ground up. [Sign up](https://cloud.google.com/free/) for $300 in free credits. +* Google Cloud Platform SDK: Follow the Google Cloud SDK [documentation](https://cloud.google.com/sdk/) to install and configure the `gcloud` command + line utility. Make sure to set a default compute region and compute zone. +* The [kubectl](https://kubernetes.io/docs/tasks/tools/install-kubectl/) command line utility is used to interact with the Kubernetes + API Server. +* Linux or Mac environment with Python 3 + +## Provisioning Compute Resources + +Kubernetes requires a set of machines to host the Kubernetes control plane and the worker nodes where containers are ultimately run. In this lab you will provision the compute resources required for running a secure and highly available Kubernetes cluster across a single [compute zone](https://cloud.google.com/compute/docs/regions-zones/regions-zones). + +### Networking + +The Kubernetes [networking model](https://kubernetes.io/docs/concepts/cluster-administration/networking/#kubernetes-model) assumes a flat network in which containers and nodes can communicate with each other. In cases where this is not desired [network policies](https://kubernetes.io/docs/concepts/services-networking/network-policies/) can limit how groups of containers are allowed to communicate with each other and external network endpoints. + +> Setting up network policies is out of scope for this tutorial. + +#### Virtual Private Cloud Network + +In this section a dedicated [Virtual Private Cloud](https://cloud.google.com/compute/docs/networks-and-firewalls#networks) (VPC) network will be setup to host the Kubernetes cluster. + +Create the `kubernetes-the-kubespray-way` custom VPC network: + +```ShellSession +gcloud compute networks create kubernetes-the-kubespray-way --subnet-mode custom +``` + +A [subnet](https://cloud.google.com/compute/docs/vpc/#vpc_networks_and_subnets) must be provisioned with an IP address range large enough to assign a private IP address to each node in the Kubernetes cluster. + +Create the `kubernetes` subnet in the `kubernetes-the-kubespray-way` VPC network: + +```ShellSession +gcloud compute networks subnets create kubernetes \ + --network kubernetes-the-kubespray-way \ + --range 10.240.0.0/24 + ``` + +> The `10.240.0.0/24` IP address range can host up to 254 compute instances. + +#### Firewall Rules + +Create a firewall rule that allows internal communication across all protocols. +It is important to note that the vxlan protocol has to be allowed in order for +the calico (see later) networking plugin to work. + +```ShellSession +gcloud compute firewall-rules create kubernetes-the-kubespray-way-allow-internal \ + --allow tcp,udp,icmp,vxlan \ + --network kubernetes-the-kubespray-way \ + --source-ranges 10.240.0.0/24 +``` + +Create a firewall rule that allows external SSH, ICMP, and HTTPS: + +```ShellSession +gcloud compute firewall-rules create kubernetes-the-kubespray-way-allow-external \ + --allow tcp:80,tcp:6443,tcp:443,tcp:22,icmp \ + --network kubernetes-the-kubespray-way \ + --source-ranges 0.0.0.0/0 +``` + +It is not feasible to restrict the firewall to a specific IP address from +where you are accessing the cluster as the nodes also communicate over the public internet and would otherwise run into +this firewall. Technically you could limit the firewall to the (fixed) IP +addresses of the cluster nodes and the remote IP addresses for accessing the +cluster. + +### Compute Instances + +The compute instances in this lab will be provisioned using [Ubuntu Server](https://www.ubuntu.com/server) 18.04. +Each compute instance will be provisioned with a fixed private IP address and + a public IP address (that can be fixed - see [guide](https://cloud.google.com/compute/docs/ip-addresses/reserve-static-external-ip-address)). +Using fixed public IP addresses has the advantage that our cluster node +configuration does not need to be updated with new public IP addresses every +time the machines are shut down and later on restarted. + +Create three compute instances which will host the Kubernetes control plane: + +```ShellSession +for i in 0 1 2; do + gcloud compute instances create controller-${i} \ + --async \ + --boot-disk-size 200GB \ + --can-ip-forward \ + --image-family ubuntu-1804-lts \ + --image-project ubuntu-os-cloud \ + --machine-type e2-standard-2 \ + --private-network-ip 10.240.0.1${i} \ + --scopes compute-rw,storage-ro,service-management,service-control,logging-write,monitoring \ + --subnet kubernetes \ + --tags kubernetes-the-kubespray-way,controller +done +``` + +> Do not forget to fix the IP addresses if you plan on re-using the cluster +after temporarily shutting down the VMs - see [guide](https://cloud.google.com/compute/docs/ip-addresses/reserve-static-external-ip-address) + +Create three compute instances which will host the Kubernetes worker nodes: + +```ShellSession +for i in 0 1 2; do + gcloud compute instances create worker-${i} \ + --async \ + --boot-disk-size 200GB \ + --can-ip-forward \ + --image-family ubuntu-1804-lts \ + --image-project ubuntu-os-cloud \ + --machine-type e2-standard-2 \ + --private-network-ip 10.240.0.2${i} \ + --scopes compute-rw,storage-ro,service-management,service-control,logging-write,monitoring \ + --subnet kubernetes \ + --tags kubernetes-the-kubespray-way,worker +done +``` + +> Do not forget to fix the IP addresses if you plan on re-using the cluster +after temporarily shutting down the VMs - see [guide](https://cloud.google.com/compute/docs/ip-addresses/reserve-static-external-ip-address) + +List the compute instances in your default compute zone: + +```ShellSession +gcloud compute instances list --filter="tags.items=kubernetes-the-kubespray-way" +``` + +> Output + +```ShellSession +NAME ZONE MACHINE_TYPE PREEMPTIBLE INTERNAL_IP EXTERNAL_IP STATUS +controller-0 us-west1-c e2-standard-2 10.240.0.10 XX.XX.XX.XXX RUNNING +controller-1 us-west1-c e2-standard-2 10.240.0.11 XX.XXX.XXX.XX RUNNING +controller-2 us-west1-c e2-standard-2 10.240.0.12 XX.XXX.XX.XXX RUNNING +worker-0 us-west1-c e2-standard-2 10.240.0.20 XX.XX.XXX.XXX RUNNING +worker-1 us-west1-c e2-standard-2 10.240.0.21 XX.XX.XX.XXX RUNNING +worker-2 us-west1-c e2-standard-2 10.240.0.22 XX.XXX.XX.XX RUNNING +``` + +### Configuring SSH Access + +Kubespray is relying on SSH to configure the controller and worker instances. + +Test SSH access to the `controller-0` compute instance: + +```ShellSession +IP_CONTROLLER_0=$(gcloud compute instances list --filter="tags.items=kubernetes-the-kubespray-way AND name:controller-0" --format="value(EXTERNAL_IP)") +USERNAME=$(whoami) +ssh $USERNAME@$IP_CONTROLLER_0 +``` + +If this is your first time connecting to a compute instance SSH keys will be +generated for you. In this case you will need to enter a passphrase at the +prompt to continue. + +> If you get a 'Remote host identification changed!' warning, you probably +already connected to that IP address in the past with another host key. You +can remove the old host key by running `ssh-keygen -R $IP_CONTROLLER_0` + +Please repeat this procedure for all the controller and worker nodes, to +ensure that SSH access is properly functioning for all nodes. + +## Set-up Kubespray + +The following set of instruction is based on the [Quick Start](https://github.com/kubernetes-sigs/kubespray) but slightly altered for our +set-up. + +As Ansible is a python application, we will create a fresh virtual +environment to install the dependencies for the Kubespray playbook: + +```ShellSession +python3 -m venv venv +source venv/bin/activate +``` + +Next, we will git clone the Kubespray code into our working directory: + +```ShellSession +git clone https://github.com/kubernetes-sigs/kubespray.git +cd kubespray +git checkout release-2.17 +``` + +Now we need to install the dependencies for Ansible to run the Kubespray +playbook: + +```ShellSession +pip install -r requirements.txt +``` + +Copy ``inventory/sample`` as ``inventory/mycluster``: + +```ShellSession +cp -rfp inventory/sample inventory/mycluster +``` + +Update Ansible inventory file with inventory builder: + +```ShellSession +declare -a IPS=($(gcloud compute instances list --filter="tags.items=kubernetes-the-kubespray-way" --format="value(EXTERNAL_IP)" | tr '\n' ' ')) +CONFIG_FILE=inventory/mycluster/hosts.yaml python3 contrib/inventory_builder/inventory.py ${IPS[@]} +``` + +Open the generated `inventory/mycluster/hosts.yaml` file and adjust it so +that controller-0, controller-1 and controller-2 are control plane nodes and +worker-0, worker-1 and worker-2 are worker nodes. Also update the `ip` to the respective local VPC IP and +remove the `access_ip`. + +The main configuration for the cluster is stored in +`inventory/mycluster/group_vars/k8s_cluster/k8s_cluster.yml`. In this file we + will update the `supplementary_addresses_in_ssl_keys` with a list of the IP + addresses of the controller nodes. In that way we can access the + kubernetes API server as an administrator from outside the VPC network. You + can also see that the `kube_network_plugin` is by default set to 'calico'. + If you set this to 'cloud', it did not work on GCP at the time of testing. + +Kubespray also offers to easily enable popular kubernetes add-ons. You can +modify the +list of add-ons in `inventory/mycluster/group_vars/k8s_cluster/addons.yml`. +Let's enable the metrics server as this is a crucial monitoring element for +the kubernetes cluster, just change the 'false' to 'true' for +`metrics_server_enabled`. + +Now we will deploy the configuration: + +```ShellSession +ansible-playbook -i inventory/mycluster/hosts.yaml -u $USERNAME -b -v --private-key=~/.ssh/id_rsa cluster.yml +``` + +Ansible will now execute the playbook, this can take up to 20 minutes. + +## Access the kubernetes cluster + +We will leverage a kubeconfig file from one of the controller nodes to access + the cluster as administrator from our local workstation. + +> In this simplified set-up, we did not include a load balancer that usually sits on top of the three controller nodes for a high available API server endpoint. In this simplified tutorial we connect directly to one of the three controllers. + +First, we need to edit the permission of the kubeconfig file on one of the +controller nodes: + +```ShellSession +ssh $USERNAME@$IP_CONTROLLER_0 +USERNAME=$(whoami) +sudo chown -R $USERNAME:$USERNAME /etc/kubernetes/admin.conf +exit +``` + +Now we will copy over the kubeconfig file: + +```ShellSession +scp $USERNAME@$IP_CONTROLLER_0:/etc/kubernetes/admin.conf kubespray-do.conf +``` + +This kubeconfig file uses the internal IP address of the controller node to +access the API server. This kubeconfig file will thus not work of from +outside the VPC network. We will need to change the API server IP address +to the controller node his external IP address. The external IP address will be +accepted in the +TLS negotiation as we added the controllers external IP addresses in the SSL +certificate configuration. +Open the file and modify the server IP address from the local IP to the +external IP address of controller-0, as stored in $IP_CONTROLLER_0. + +> Example + +```ShellSession +apiVersion: v1 +clusters: +- cluster: + certificate-authority-data: XXX + server: https://35.205.205.80:6443 + name: cluster.local +... +``` + +Now, we load the configuration for `kubectl`: + +```ShellSession +export KUBECONFIG=$PWD/kubespray-do.conf +``` + +We should be all set to communicate with our cluster from our local workstation: + +```ShellSession +kubectl get nodes +``` + +> Output + +```ShellSession +NAME STATUS ROLES AGE VERSION +controller-0 Ready master 47m v1.17.9 +controller-1 Ready master 46m v1.17.9 +controller-2 Ready master 46m v1.17.9 +worker-0 Ready 45m v1.17.9 +worker-1 Ready 45m v1.17.9 +worker-2 Ready 45m v1.17.9 +``` + +## Smoke tests + +### Metrics + +Verify if the metrics server addon was correctly installed and works: + +```ShellSession +kubectl top nodes +``` + +> Output + +```ShellSession +NAME CPU(cores) CPU% MEMORY(bytes) MEMORY% +controller-0 191m 10% 1956Mi 26% +controller-1 190m 10% 1828Mi 24% +controller-2 182m 10% 1839Mi 24% +worker-0 87m 4% 1265Mi 16% +worker-1 102m 5% 1268Mi 16% +worker-2 108m 5% 1299Mi 17% +``` + +Please note that metrics might not be available at first and need a couple of + minutes before you can actually retrieve them. + +### Network + +Let's verify if the network layer is properly functioning and pods can reach +each other: + +```ShellSession +kubectl run myshell1 -it --rm --image busybox -- sh +hostname -i +# launch myshell2 in separate terminal (see next code block) and ping the hostname of myshell2 +ping +``` + +```ShellSession +kubectl run myshell2 -it --rm --image busybox -- sh +hostname -i +ping +``` + +> Output + +```ShellSession +PING 10.233.108.2 (10.233.108.2): 56 data bytes +64 bytes from 10.233.108.2: seq=0 ttl=62 time=2.876 ms +64 bytes from 10.233.108.2: seq=1 ttl=62 time=0.398 ms +64 bytes from 10.233.108.2: seq=2 ttl=62 time=0.378 ms +^C +--- 10.233.108.2 ping statistics --- +3 packets transmitted, 3 packets received, 0% packet loss +round-trip min/avg/max = 0.378/1.217/2.876 ms +``` + +### Deployments + +In this section you will verify the ability to create and manage [Deployments](https://kubernetes.io/docs/concepts/workloads/controllers/deployment/). + +Create a deployment for the [nginx](https://nginx.org/en/) web server: + +```ShellSession +kubectl create deployment nginx --image=nginx +``` + +List the pod created by the `nginx` deployment: + +```ShellSession +kubectl get pods -l app=nginx +``` + +> Output + +```ShellSession +NAME READY STATUS RESTARTS AGE +nginx-86c57db685-bmtt8 1/1 Running 0 18s +``` + +#### Port Forwarding + +In this section you will verify the ability to access applications remotely using [port forwarding](https://kubernetes.io/docs/tasks/access-application-cluster/port-forward-access-application-cluster/). + +Retrieve the full name of the `nginx` pod: + +```ShellSession +POD_NAME=$(kubectl get pods -l app=nginx -o jsonpath="{.items[0].metadata.name}") +``` + +Forward port `8080` on your local machine to port `80` of the `nginx` pod: + +```ShellSession +kubectl port-forward $POD_NAME 8080:80 +``` + +> Output + +```ShellSession +Forwarding from 127.0.0.1:8080 -> 80 +Forwarding from [::1]:8080 -> 80 +``` + +In a new terminal make an HTTP request using the forwarding address: + +```ShellSession +curl --head http://127.0.0.1:8080 +``` + +> Output + +```ShellSession +HTTP/1.1 200 OK +Server: nginx/1.19.1 +Date: Thu, 13 Aug 2020 11:12:04 GMT +Content-Type: text/html +Content-Length: 612 +Last-Modified: Tue, 07 Jul 2020 15:52:25 GMT +Connection: keep-alive +ETag: "5f049a39-264" +Accept-Ranges: bytes +``` + +Switch back to the previous terminal and stop the port forwarding to the `nginx` pod: + +```ShellSession +Forwarding from 127.0.0.1:8080 -> 80 +Forwarding from [::1]:8080 -> 80 +Handling connection for 8080 +^C +``` + +#### Logs + +In this section you will verify the ability to [retrieve container logs](https://kubernetes.io/docs/concepts/cluster-administration/logging/). + +Print the `nginx` pod logs: + +```ShellSession +kubectl logs $POD_NAME +``` + +> Output + +```ShellSession +... +127.0.0.1 - - [13/Aug/2020:11:12:04 +0000] "HEAD / HTTP/1.1" 200 0 "-" "curl/7.64.1" "-" +``` + +#### Exec + +In this section you will verify the ability to [execute commands in a container](https://kubernetes.io/docs/tasks/debug/debug-application/get-shell-running-container/#running-individual-commands-in-a-container). + +Print the nginx version by executing the `nginx -v` command in the `nginx` container: + +```ShellSession +kubectl exec -ti $POD_NAME -- nginx -v +``` + +> Output + +```ShellSession +nginx version: nginx/1.19.1 +``` + +### Kubernetes services + +#### Expose outside the cluster + +In this section you will verify the ability to expose applications using a [Service](https://kubernetes.io/docs/concepts/services-networking/service/). + +Expose the `nginx` deployment using a [NodePort](https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport) service: + +```ShellSession +kubectl expose deployment nginx --port 80 --type NodePort +``` + +> The LoadBalancer service type can not be used because your cluster is not configured with [cloud provider integration](https://kubernetes.io/docs/getting-started-guides/scratch/#cloud-provider). Setting up cloud provider integration is out of scope for this tutorial. + +Retrieve the node port assigned to the `nginx` service: + +```ShellSession +NODE_PORT=$(kubectl get svc nginx \ + --output=jsonpath='{range .spec.ports[0]}{.nodePort}') +``` + +Create a firewall rule that allows remote access to the `nginx` node port: + +```ShellSession +gcloud compute firewall-rules create kubernetes-the-kubespray-way-allow-nginx-service \ + --allow=tcp:${NODE_PORT} \ + --network kubernetes-the-kubespray-way +``` + +Retrieve the external IP address of a worker instance: + +```ShellSession +EXTERNAL_IP=$(gcloud compute instances describe worker-0 \ + --format 'value(networkInterfaces[0].accessConfigs[0].natIP)') +``` + +Make an HTTP request using the external IP address and the `nginx` node port: + +```ShellSession +curl -I http://${EXTERNAL_IP}:${NODE_PORT} +``` + +> Output + +```ShellSession +HTTP/1.1 200 OK +Server: nginx/1.19.1 +Date: Thu, 13 Aug 2020 11:15:02 GMT +Content-Type: text/html +Content-Length: 612 +Last-Modified: Tue, 07 Jul 2020 15:52:25 GMT +Connection: keep-alive +ETag: "5f049a39-264" +Accept-Ranges: bytes +``` + +#### Local DNS + +We will now also verify that kubernetes built-in DNS works across namespaces. +Create a namespace: + +```ShellSession +kubectl create namespace dev +``` + +Create an nginx deployment and expose it within the cluster: + +```ShellSession +kubectl create deployment nginx --image=nginx -n dev +kubectl expose deployment nginx --port 80 --type ClusterIP -n dev +``` + +Run a temporary container to see if we can reach the service from the default +namespace: + +```ShellSession +kubectl run curly -it --rm --image curlimages/curl:7.70.0 -- /bin/sh +curl --head http://nginx.dev:80 +``` + +> Output + +```ShellSession +HTTP/1.1 200 OK +Server: nginx/1.19.1 +Date: Thu, 13 Aug 2020 11:15:59 GMT +Content-Type: text/html +Content-Length: 612 +Last-Modified: Tue, 07 Jul 2020 15:52:25 GMT +Connection: keep-alive +ETag: "5f049a39-264" +Accept-Ranges: bytes +``` + +Type `exit` to leave the shell. + +## Cleaning Up + +### Kubernetes resources + +Delete the dev namespace, the nginx deployment and service: + +```ShellSession +kubectl delete namespace dev +kubectl delete deployment nginx +kubectl delete svc/nginx +``` + +### Kubernetes state + +Note: you can skip this step if you want to entirely remove the machines. + +If you want to keep the VMs and just remove the cluster state, you can simply + run another Ansible playbook: + +```ShellSession +ansible-playbook -i inventory/mycluster/hosts.yaml -u $USERNAME -b -v --private-key=~/.ssh/id_rsa reset.yml +``` + +Resetting the cluster to the VMs original state usually takes about a couple +of minutes. + +### Compute instances + +Delete the controller and worker compute instances: + +```ShellSession +gcloud -q compute instances delete \ + controller-0 controller-1 controller-2 \ + worker-0 worker-1 worker-2 \ + --zone $(gcloud config get-value compute/zone) + ``` + + +### Network + + +Delete the fixed IP addresses (assuming you named them equal to the VM names), +if any: + +```ShellSession +gcloud -q compute addresses delete controller-0 controller-1 controller-2 \ + worker-0 worker-1 worker-2 +``` + +Delete the `kubernetes-the-kubespray-way` firewall rules: + +```ShellSession +gcloud -q compute firewall-rules delete \ + kubernetes-the-kubespray-way-allow-nginx-service \ + kubernetes-the-kubespray-way-allow-internal \ + kubernetes-the-kubespray-way-allow-external +``` + +Delete the `kubernetes-the-kubespray-way` network VPC: + +```ShellSession +gcloud -q compute networks subnets delete kubernetes +gcloud -q compute networks delete kubernetes-the-kubespray-way +``` diff --git a/kubespray/docs/test_cases.md b/kubespray/docs/test_cases.md new file mode 100644 index 0000000..1fdce68 --- /dev/null +++ b/kubespray/docs/test_cases.md @@ -0,0 +1,33 @@ +# Node Layouts + +There are four node layout types: `default`, `separate`, `ha`, and `scale`. + +`default` is a non-HA two nodes setup with one separate `kube_node` +and the `etcd` group merged with the `kube_control_plane`. + +`separate` layout is when there is only node of each type, which includes + a kube_control_plane, kube_node, and etcd cluster member. + +`ha` layout consists of two etcd nodes, two control planes and a single worker node, +with role intersection. + +`scale` layout can be combined with above layouts (`ha-scale`, `separate-scale`). It includes 200 fake hosts +in the Ansible inventory. This helps test TLS certificate generation at scale +to prevent regressions and profile certain long-running tasks. These nodes are +never actually deployed, but certificates are generated for them. + +Note, the canal network plugin deploys flannel as well plus calico policy controller. + +## Test cases + +The [CI Matrix](/docs/ci.md) displays OS, Network Plugin and Container Manager tested. + +All tests are breakdown into 3 "stages" ("Stage" means a build step of the build pipeline) as follows: + +- _unit_tests_: Linting, markdown, vagrant & terraform validation etc... +- _part1_: Molecule and AIO tests +- _part2_: Standard tests with different layouts and OS/Runtime/Network +- _part3_: Upgrade jobs, terraform jobs and recover control plane tests +- _special_: Other jobs (manuals) + +The steps are ordered as `unit_tests->part1->part2->part3->special`. diff --git a/kubespray/docs/uoslinux.md b/kubespray/docs/uoslinux.md new file mode 100644 index 0000000..1078389 --- /dev/null +++ b/kubespray/docs/uoslinux.md @@ -0,0 +1,9 @@ +# UOS Linux + +UOS Linux(UnionTech OS Server 20) is supported with docker and containerd runtimes. + +**Note:** that UOS Linux is not currently covered in kubespray CI and +support for it is currently considered experimental. + +There are no special considerations for using UOS Linux as the target OS +for Kubespray deployments. diff --git a/kubespray/docs/upgrades.md b/kubespray/docs/upgrades.md new file mode 100644 index 0000000..52dccba --- /dev/null +++ b/kubespray/docs/upgrades.md @@ -0,0 +1,418 @@ +# Upgrading Kubernetes in Kubespray + +Kubespray handles upgrades the same way it handles initial deployment. That is to +say that each component is laid down in a fixed order. + +You can also individually control versions of components by explicitly defining their +versions. Here are all version vars for each component: + +* docker_version +* docker_containerd_version (relevant when `container_manager` == `docker`) +* containerd_version (relevant when `container_manager` == `containerd`) +* kube_version +* etcd_version +* calico_version +* calico_cni_version +* weave_version +* flannel_version +* kubedns_version + +> **Warning** +> [Attempting to upgrade from an older release straight to the latest release is unsupported and likely to break something](https://github.com/kubernetes-sigs/kubespray/issues/3849#issuecomment-451386515) + +See [Multiple Upgrades](#multiple-upgrades) for how to upgrade from older Kubespray release to the latest release + +## Unsafe upgrade example + +If you wanted to upgrade just kube_version from v1.18.10 to v1.19.7, you could +deploy the following way: + +```ShellSession +ansible-playbook cluster.yml -i inventory/sample/hosts.ini -e kube_version=v1.18.10 -e upgrade_cluster_setup=true +``` + +And then repeat with v1.19.7 as kube_version: + +```ShellSession +ansible-playbook cluster.yml -i inventory/sample/hosts.ini -e kube_version=v1.19.7 -e upgrade_cluster_setup=true +``` + +The var ```-e upgrade_cluster_setup=true``` is needed to be set in order to migrate the deploys of e.g kube-apiserver inside the cluster immediately which is usually only done in the graceful upgrade. (Refer to [#4139](https://github.com/kubernetes-sigs/kubespray/issues/4139) and [#4736](https://github.com/kubernetes-sigs/kubespray/issues/4736)) + +## Graceful upgrade + +Kubespray also supports cordon, drain and uncordoning of nodes when performing +a cluster upgrade. There is a separate playbook used for this purpose. It is +important to note that upgrade-cluster.yml can only be used for upgrading an +existing cluster. That means there must be at least 1 kube_control_plane already +deployed. + +```ShellSession +ansible-playbook upgrade-cluster.yml -b -i inventory/sample/hosts.ini -e kube_version=v1.19.7 +``` + +After a successful upgrade, the Server Version should be updated: + +```ShellSession +$ kubectl version +Client Version: version.Info{Major:"1", Minor:"19", GitVersion:"v1.19.7", GitCommit:"1dd5338295409edcfff11505e7bb246f0d325d15", GitTreeState:"clean", BuildDate:"2021-01-13T13:23:52Z", GoVersion:"go1.15.5", Compiler:"gc", Platform:"linux/amd64"} +Server Version: version.Info{Major:"1", Minor:"19", GitVersion:"v1.19.7", GitCommit:"1dd5338295409edcfff11505e7bb246f0d325d15", GitTreeState:"clean", BuildDate:"2021-01-13T13:15:20Z", GoVersion:"go1.15.5", Compiler:"gc", Platform:"linux/amd64"} +``` + +### Pausing the upgrade + +If you want to manually control the upgrade procedure, you can set some variables to pause the upgrade playbook. Pausing *before* upgrading each upgrade may be useful for inspecting pods running on that node, or performing manual actions on the node: + +* `upgrade_node_confirm: true` - This will pause the playbook execution prior to upgrading each node. The play will resume when manually approved by typing "yes" at the terminal. +* `upgrade_node_pause_seconds: 60` - This will pause the playbook execution for 60 seconds prior to upgrading each node. The play will resume automatically after 60 seconds. + +Pausing *after* upgrading each node may be useful for rebooting the node to apply kernel updates, or testing the still-cordoned node: + +* `upgrade_node_post_upgrade_confirm: true` - This will pause the playbook execution after upgrading each node, but before the node is uncordoned. The play will resume when manually approved by typing "yes" at the terminal. +* `upgrade_node_post_upgrade_pause_seconds: 60` - This will pause the playbook execution for 60 seconds after upgrading each node, but before the node is uncordoned. The play will resume automatically after 60 seconds. + +## Node-based upgrade + +If you don't want to upgrade all nodes in one run, you can use `--limit` [patterns](https://docs.ansible.com/ansible/latest/user_guide/intro_patterns.html#patterns-and-ansible-playbook-flags). + +Before using `--limit` run playbook `facts.yml` without the limit to refresh facts cache for all nodes: + +```ShellSession +ansible-playbook facts.yml -b -i inventory/sample/hosts.ini +``` + +After this upgrade control plane and etcd groups [#5147](https://github.com/kubernetes-sigs/kubespray/issues/5147): + +```ShellSession +ansible-playbook upgrade-cluster.yml -b -i inventory/sample/hosts.ini -e kube_version=v1.20.7 --limit "kube_control_plane:etcd" +``` + +Now you can upgrade other nodes in any order and quantity: + +```ShellSession +ansible-playbook upgrade-cluster.yml -b -i inventory/sample/hosts.ini -e kube_version=v1.20.7 --limit "node4:node6:node7:node12" +ansible-playbook upgrade-cluster.yml -b -i inventory/sample/hosts.ini -e kube_version=v1.20.7 --limit "node5*" +``` + +## Multiple upgrades + +> **Warning** +> [Do not skip releases when upgrading--upgrade by one tag at a time.](https://github.com/kubernetes-sigs/kubespray/issues/3849#issuecomment-451386515) + +For instance, if you're on v2.6.0, then check out v2.7.0, run the upgrade, check out the next tag, and run the next upgrade, etc. + +Assuming you don't explicitly define a kubernetes version in your k8s_cluster.yml, you simply check out the next tag and run the upgrade-cluster.yml playbook + +* If you do define kubernetes version in your inventory (e.g. group_vars/k8s_cluster.yml) then either make sure to update it before running upgrade-cluster, or specify the new version you're upgrading to: `ansible-playbook -i inventory/mycluster/hosts.ini -b upgrade-cluster.yml -e kube_version=v1.11.3` + + Otherwise, the upgrade will leave your cluster at the same k8s version defined in your inventory vars. + +The below example shows taking a cluster that was set up for v2.6.0 up to v2.10.0 + +```ShellSession +$ kubectl get node +NAME STATUS ROLES AGE VERSION +apollo Ready master,node 1h v1.10.4 +boomer Ready master,node 42m v1.10.4 +caprica Ready master,node 42m v1.10.4 + +$ git describe --tags +v2.6.0 + +$ git tag +... +v2.6.0 +v2.7.0 +v2.8.0 +v2.8.1 +v2.8.2 +... + +$ git checkout v2.7.0 +Previous HEAD position was 8b3ce6e4 bump upgrade tests to v2.5.0 commit (#3087) +HEAD is now at 05dabb7e Fix Bionic networking restart error #3430 (#3431) + +# NOTE: May need to `pip3 install -r requirements.txt` when upgrading. + +ansible-playbook -i inventory/mycluster/hosts.ini -b upgrade-cluster.yml + +... + +$ kubectl get node +NAME STATUS ROLES AGE VERSION +apollo Ready master,node 1h v1.11.3 +boomer Ready master,node 1h v1.11.3 +caprica Ready master,node 1h v1.11.3 + +$ git checkout v2.8.0 +Previous HEAD position was 05dabb7e Fix Bionic networking restart error #3430 (#3431) +HEAD is now at 9051aa52 Fix ubuntu-contiv test failed (#3808) +``` + +> **Note** +> Review changes between the sample inventory and your inventory when upgrading versions. + +Some deprecations between versions that mean you can't just upgrade straight from 2.7.0 to 2.8.0 if you started with the sample inventory. + +In this case, I set "kubeadm_enabled" to false, knowing that it is deprecated and removed by 2.9.0, to delay converting the cluster to kubeadm as long as I could. + +```ShellSession +$ ansible-playbook -i inventory/mycluster/hosts.ini -b upgrade-cluster.yml +... + "msg": "DEPRECATION: non-kubeadm deployment is deprecated from v2.9. Will be removed in next release." +... +Are you sure you want to deploy cluster using the deprecated non-kubeadm mode. (output is hidden): +yes +... + +$ kubectl get node +NAME STATUS ROLES AGE VERSION +apollo Ready master,node 114m v1.12.3 +boomer Ready master,node 114m v1.12.3 +caprica Ready master,node 114m v1.12.3 + +$ git checkout v2.8.1 +Previous HEAD position was 9051aa52 Fix ubuntu-contiv test failed (#3808) +HEAD is now at 2ac1c756 More Feature/2.8 backports for 2.8.1 (#3911) + +$ ansible-playbook -i inventory/mycluster/hosts.ini -b upgrade-cluster.yml +... + "msg": "DEPRECATION: non-kubeadm deployment is deprecated from v2.9. Will be removed in next release." +... +Are you sure you want to deploy cluster using the deprecated non-kubeadm mode. (output is hidden): +yes +... + +$ kubectl get node +NAME STATUS ROLES AGE VERSION +apollo Ready master,node 2h36m v1.12.4 +boomer Ready master,node 2h36m v1.12.4 +caprica Ready master,node 2h36m v1.12.4 + +$ git checkout v2.8.2 +Previous HEAD position was 2ac1c756 More Feature/2.8 backports for 2.8.1 (#3911) +HEAD is now at 4167807f Upgrade to 1.12.5 (#4066) + +$ ansible-playbook -i inventory/mycluster/hosts.ini -b upgrade-cluster.yml +... + "msg": "DEPRECATION: non-kubeadm deployment is deprecated from v2.9. Will be removed in next release." +... +Are you sure you want to deploy cluster using the deprecated non-kubeadm mode. (output is hidden): +yes +... + +$ kubectl get node +NAME STATUS ROLES AGE VERSION +apollo Ready master,node 3h3m v1.12.5 +boomer Ready master,node 3h3m v1.12.5 +caprica Ready master,node 3h3m v1.12.5 + +$ git checkout v2.8.3 +Previous HEAD position was 4167807f Upgrade to 1.12.5 (#4066) +HEAD is now at ea41fc5e backport cve-2019-5736 to release-2.8 (#4234) + +$ ansible-playbook -i inventory/mycluster/hosts.ini -b upgrade-cluster.yml +... + "msg": "DEPRECATION: non-kubeadm deployment is deprecated from v2.9. Will be removed in next release." +... +Are you sure you want to deploy cluster using the deprecated non-kubeadm mode. (output is hidden): +yes +... + +$ kubectl get node +NAME STATUS ROLES AGE VERSION +apollo Ready master,node 5h18m v1.12.5 +boomer Ready master,node 5h18m v1.12.5 +caprica Ready master,node 5h18m v1.12.5 + +$ git checkout v2.8.4 +Previous HEAD position was ea41fc5e backport cve-2019-5736 to release-2.8 (#4234) +HEAD is now at 3901480b go to k8s 1.12.7 (#4400) + +$ ansible-playbook -i inventory/mycluster/hosts.ini -b upgrade-cluster.yml +... + "msg": "DEPRECATION: non-kubeadm deployment is deprecated from v2.9. Will be removed in next release." +... +Are you sure you want to deploy cluster using the deprecated non-kubeadm mode. (output is hidden): +yes +... + +$ kubectl get node +NAME STATUS ROLES AGE VERSION +apollo Ready master,node 5h37m v1.12.7 +boomer Ready master,node 5h37m v1.12.7 +caprica Ready master,node 5h37m v1.12.7 + +$ git checkout v2.8.5 +Previous HEAD position was 3901480b go to k8s 1.12.7 (#4400) +HEAD is now at 6f97687d Release 2.8 robust san handling (#4478) + +$ ansible-playbook -i inventory/mycluster/hosts.ini -b upgrade-cluster.yml +... + "msg": "DEPRECATION: non-kubeadm deployment is deprecated from v2.9. Will be removed in next release." +... +Are you sure you want to deploy cluster using the deprecated non-kubeadm mode. (output is hidden): +yes +... + +$ kubectl get node +NAME STATUS ROLES AGE VERSION +apollo Ready master,node 5h45m v1.12.7 +boomer Ready master,node 5h45m v1.12.7 +caprica Ready master,node 5h45m v1.12.7 + +$ git checkout v2.9.0 +Previous HEAD position was 6f97687d Release 2.8 robust san handling (#4478) +HEAD is now at a4e65c7c Upgrade to Ansible >2.7.0 (#4471) +``` + +> **Warning** +> IMPORTANT: Some variable formats changed in the k8s_cluster.yml between 2.8.5 and 2.9.0 + +If you do not keep your inventory copy up to date, **your upgrade will fail** and your first master will be left non-functional until fixed and re-run. + +It is at this point the cluster was upgraded from non-kubeadm to kubeadm as per the deprecation warning. + +```ShellSession +ansible-playbook -i inventory/mycluster/hosts.ini -b upgrade-cluster.yml + +... + +$ kubectl get node +NAME STATUS ROLES AGE VERSION +apollo Ready master,node 6h54m v1.13.5 +boomer Ready master,node 6h55m v1.13.5 +caprica Ready master,node 6h54m v1.13.5 + +# Watch out: 2.10.0 is hiding between 2.1.2 and 2.2.0 + +$ git tag +... +v2.1.0 +v2.1.1 +v2.1.2 +v2.10.0 +v2.2.0 +... + +$ git checkout v2.10.0 +Previous HEAD position was a4e65c7c Upgrade to Ansible >2.7.0 (#4471) +HEAD is now at dcd9c950 Add etcd role dependency on kube user to avoid etcd role failure when running scale.yml with a fresh node. (#3240) (#4479) + +ansible-playbook -i inventory/mycluster/hosts.ini -b upgrade-cluster.yml + +... + +$ kubectl get node +NAME STATUS ROLES AGE VERSION +apollo Ready master,node 7h40m v1.14.1 +boomer Ready master,node 7h40m v1.14.1 +caprica Ready master,node 7h40m v1.14.1 + + +``` + +## Upgrading to v2.19 + +`etcd_kubeadm_enabled` is being deprecated at v2.19. The same functionality is achievable by setting `etcd_deployment_type` to `kubeadm`. +Deploying etcd using kubeadm is experimental and is only available for either new or deployments where `etcd_kubeadm_enabled` was set to `true` while deploying the cluster. + +From 2.19 and onward `etcd_deployment_type` variable will be placed in `group_vars/all/etcd.yml` instead of `group_vars/etcd.yml`, due to scope issues. +The placement of the variable is only important for `etcd_deployment_type: kubeadm` right now. However, since this might change in future updates, it is recommended to move the variable. + +Upgrading is straightforward; no changes are required if `etcd_kubeadm_enabled` was not set to `true` when deploying. + +If you have a cluster where `etcd` was deployed using `kubeadm`, you will need to remove `etcd_kubeadm_enabled` the variable. Then move `etcd_deployment_type` variable from `group_vars/etcd.yml` to `group_vars/all/etcd.yml` due to scope issues and set `etcd_deployment_type` to `kubeadm`. + +## Upgrade order + +As mentioned above, components are upgraded in the order in which they were +installed in the Ansible playbook. The order of component installation is as +follows: + +* Docker +* Containerd +* etcd +* kubelet and kube-proxy +* network_plugin (such as Calico or Weave) +* kube-apiserver, kube-scheduler, and kube-controller-manager +* Add-ons (such as KubeDNS) + +### Component-based upgrades + +A deployer may want to upgrade specific components in order to minimize risk +or save time. This strategy is not covered by CI as of this writing, so it is +not guaranteed to work. + +These commands are useful only for upgrading fully-deployed, healthy, existing +hosts. This will definitely not work for undeployed or partially deployed +hosts. + +Upgrade docker: + +```ShellSession +ansible-playbook -b -i inventory/sample/hosts.ini cluster.yml --tags=docker +``` + +Upgrade etcd: + +```ShellSession +ansible-playbook -b -i inventory/sample/hosts.ini cluster.yml --tags=etcd +``` + +Upgrade etcd without rotating etcd certs: + +```ShellSession +ansible-playbook -b -i inventory/sample/hosts.ini cluster.yml --tags=etcd --limit=etcd --skip-tags=etcd-secrets +``` + +Upgrade kubelet: + +```ShellSession +ansible-playbook -b -i inventory/sample/hosts.ini cluster.yml --tags=node --skip-tags=k8s-gen-certs,k8s-gen-tokens +``` + +Upgrade Kubernetes master components: + +```ShellSession +ansible-playbook -b -i inventory/sample/hosts.ini cluster.yml --tags=master +``` + +Upgrade network plugins: + +```ShellSession +ansible-playbook -b -i inventory/sample/hosts.ini cluster.yml --tags=network +``` + +Upgrade all add-ons: + +```ShellSession +ansible-playbook -b -i inventory/sample/hosts.ini cluster.yml --tags=apps +``` + +Upgrade just helm (assuming `helm_enabled` is true): + +```ShellSession +ansible-playbook -b -i inventory/sample/hosts.ini cluster.yml --tags=helm +``` + +## Migrate from Docker to Containerd + +Please note that **migrating container engines is not officially supported by Kubespray**. While this procedure can be used to migrate your cluster, it applies to one particular scenario and will likely evolve over time. At the moment, they are intended as an additional resource to provide insight into how these steps can be officially integrated into the Kubespray playbooks. + +As of Kubespray 2.18.0, containerd is already the default container engine. If you have the chance, it is advisable and safer to reset and redeploy the entire cluster with a new container engine. + +* [Migrating from Docker to Containerd](upgrades/migrate_docker2containerd.md) + +## System upgrade + +If you want to upgrade the APT or YUM packages while the nodes are cordoned, you can use: + +```ShellSession +ansible-playbook upgrade-cluster.yml -b -i inventory/sample/hosts.ini -e system_upgrade=true +``` + +Nodes will be rebooted when there are package upgrades (`system_upgrade_reboot: on-upgrade`). +This can be changed to `always` or `never`. + +Note: Downloads will happen twice unless `system_upgrade_reboot` is `never`. diff --git a/kubespray/docs/upgrades/migrate_docker2containerd.md b/kubespray/docs/upgrades/migrate_docker2containerd.md new file mode 100644 index 0000000..b444db0 --- /dev/null +++ b/kubespray/docs/upgrades/migrate_docker2containerd.md @@ -0,0 +1,106 @@ +# Migrating from Docker to Containerd + +❗MAKE SURE YOU READ BEFORE PROCEEDING❗ + +**Migrating container engines is not officially supported by Kubespray**. The following procedure covers one particular scenario and involves manual steps, along with multiple runs of `cluster.yml`. It provides no guarantees that it will actually work or that any further action is needed. Please, consider these instructions as experimental guidelines. While they can be used to migrate your cluster, they will likely evolve over time. At the moment, they are intended as an additional resource to provide insight into how these steps can be officially integrated into the Kubespray playbooks. + +As of Kubespray 2.18.0, containerd is already the default container engine. If you have the chance, it is still advisable and safer to reset and redeploy the entire cluster with a new container engine. + +Input and feedback are always appreciated. + +## Tested environment + +Nodes: Ubuntu 18.04 LTS\ +Cloud Provider: None (baremetal or VMs)\ +Kubernetes version: 1.21.5\ +Kubespray version: 2.18.0 + +## Important considerations + +If you require minimum downtime, nodes need to be cordoned and drained before being processed, one by one. If you wish to run `cluster.yml` only once and get it all done in one swoop, downtime will be significantly higher. Docker will need to be manually removed from all nodes before the playbook runs (see [#8431](https://github.com/kubernetes-sigs/kubespray/issues/8431)). For minimum downtime, the following steps will be executed multiple times, once for each node. + +Processing nodes one by one also means you will not be able to update any other cluster configuration using Kubespray before this procedure is finished and the cluster is fully migrated. + +Everything done here requires full root access to every node. + +## Migration steps + +Before you begin, adjust your inventory: + +```yaml +# Filename: k8s_cluster/k8s-cluster.yml +resolvconf_mode: host_resolvconf +container_manager: containerd + +# Filename: etcd.yml +etcd_deployment_type: host +``` + +### 1) Pick one or more nodes for processing + +It is still unclear how the order might affect this procedure. So, to be sure, it might be best to start with the control plane and etcd nodes all together, followed by each worker node individually. + +### 2) Cordon and drain the node + +... because, downtime. + +### 3) Stop docker and kubelet daemons + +```commandline +service kubelet stop +service docker stop +``` + +### 4) Uninstall docker + dependencies + +```commandline +apt-get remove -y --allow-change-held-packages containerd.io docker-ce docker-ce-cli docker-ce-rootless-extras +``` + +In some cases, there might a `pigz` missing dependency. Some image layers need this to be extracted. + +```shell +apt-get install pigz +``` + +### 5) Run `cluster.yml` playbook with `--limit` + +```commandline +ansible-playbook -i inventory/sample/hosts.ini cluster.yml --limit=NODENAME +``` + +This effectively reinstalls containerd and seems to place all config files in the right place. When this completes, kubelet will immediately pick up the new container engine and start spinning up DaemonSets and kube-system Pods. + +Optionally, if you feel confident, you can remove `/var/lib/docker` anytime after this step. + +```commandline +rm -fr /var/lib/docker +``` + +You can watch new containers using `crictl`. + +```commandline +crictl ps -a +``` + +### 6) Replace the cri-socket node annotation + +Node annotations need to be adjusted. Kubespray will not do this, but a simple kubectl is enough. + +```commandline +kubectl annotate node NODENAME --overwrite kubeadm.alpha.kubernetes.io/cri-socket=/var/run/containerd/containerd.sock +``` + +The annotation is required by kubeadm to follow through future cluster upgrades. + +### 7) Reboot the node + +Reboot, just to make sure everything restarts fresh before the node is uncordoned. + +## After thoughts + +If your cluster runs a log aggregator, like fluentd+Graylog, you will likely need to adjust collection filters and parsers. While docker generates Json logs, containerd has its own space delimited format. Example: + +```text +2020-01-10T18:10:40.01576219Z stdout F application log message... +``` diff --git a/kubespray/docs/vagrant.md b/kubespray/docs/vagrant.md new file mode 100644 index 0000000..b7f702c --- /dev/null +++ b/kubespray/docs/vagrant.md @@ -0,0 +1,164 @@ +# Vagrant + +Assuming you have Vagrant 2.0+ installed with virtualbox, libvirt/qemu or +vmware, but is untested) you should be able to launch a 3 node Kubernetes +cluster by simply running `vagrant up`. + +This will spin up 3 VMs and install kubernetes on them. +Once they are completed you can connect to any of them by running `vagrant ssh k8s-[1..3]`. + +To give an estimate of the expected duration of a provisioning run: +On a dual core i5-6300u laptop with an SSD, provisioning takes around 13 +to 15 minutes, once the container images and other files are cached. +Note that libvirt/qemu is recommended over virtualbox as it is quite a bit +faster, especially during boot-up time. + +For proper performance a minimum of 12GB RAM is recommended. +It is possible to run a 3 node cluster on a laptop with 8GB of RAM using +the default Vagrantfile, provided you have 8GB zram swap configured and +not much more than a browser and a mail client running. +If you decide to run on such a machine, then also make sure that any tmpfs +devices, that are mounted, are mostly empty and disable any swapfiles +mounted on HDD/SSD or you will be in for some serious swap-madness. +Things can get a bit sluggish during provisioning, but when that's done, +the system will actually be able to perform quite well. + +## Customize Vagrant + +You can override the default settings in the `Vagrantfile` either by +directly modifying the `Vagrantfile` or through an override file. +In the same directory as the `Vagrantfile`, create a folder called +`vagrant` and create `config.rb` file in it. +An example of how to configure this file is given below. + +## Use alternative OS for Vagrant + +By default, Vagrant uses Ubuntu 18.04 box to provision a local cluster. +You may use an alternative supported operating system for your local cluster. + +Customize `$os` variable in `Vagrantfile` or as override, e.g.,: + +```ShellSession +echo '$os = "flatcar-stable"' >> vagrant/config.rb +``` + +The supported operating systems for vagrant are defined in the `SUPPORTED_OS` +constant in the `Vagrantfile`. + +## File and image caching + +Kubespray can take quite a while to start on a laptop. To improve provisioning +speed, the variable 'download_run_once' is set. This will make kubespray +download all files and containers just once and then redistributes them to +the other nodes and as a bonus, also cache all downloads locally and re-use +them on the next provisioning run. For more information on download settings +see [download documentation](/docs/downloads.md). + +## Example use of Vagrant + +The following is an example of setting up and running kubespray using `vagrant`. +For repeated runs, you could save the script to a file in the root of the +kubespray and run it by executing `source `. + +```ShellSession +# use virtualenv to install all python requirements +VENVDIR=venv +virtualenv --python=/usr/bin/python3.7 $VENVDIR +source $VENVDIR/bin/activate +pip install -r requirements.txt + +# prepare an inventory to test with +INV=inventory/my_lab +rm -rf ${INV}.bak &> /dev/null +mv ${INV} ${INV}.bak &> /dev/null +cp -a inventory/sample ${INV} +rm -f ${INV}/hosts.ini + +# customize the vagrant environment +mkdir vagrant +cat << EOF > vagrant/config.rb +\$instance_name_prefix = "kub" +\$vm_cpus = 1 +\$num_instances = 3 +\$os = "centos-bento" +\$subnet = "10.0.20" +\$network_plugin = "flannel" +\$inventory = "$INV" +\$shared_folders = { 'temp/docker_rpms' => "/var/cache/yum/x86_64/7/docker-ce/packages" } +EOF + +# make the rpm cache +mkdir -p temp/docker_rpms + +vagrant up + +# make a copy of the downloaded docker rpm, to speed up the next provisioning run +scp kub-1:/var/cache/yum/x86_64/7/docker-ce/packages/* temp/docker_rpms/ + +# copy kubectl access configuration in place +mkdir $HOME/.kube/ &> /dev/null +ln -s $PWD/$INV/artifacts/admin.conf $HOME/.kube/config +# make the kubectl binary available +sudo ln -s $PWD/$INV/artifacts/kubectl /usr/local/bin/kubectl +#or +export PATH=$PATH:$PWD/$INV/artifacts +``` + +If a vagrant run failed and you've made some changes to fix the issue causing +the fail, here is how you would re-run ansible: + +```ShellSession +ansible-playbook -vvv -i .vagrant/provisioners/ansible/inventory/vagrant_ansible_inventory cluster.yml +``` + +If all went well, you check if it's all working as expected: + +```ShellSession +kubectl get nodes +``` + +The output should look like this: + +```ShellSession +$ kubectl get nodes +NAME STATUS ROLES AGE VERSION +kub-1 Ready control-plane,master 4m37s v1.22.5 +kub-2 Ready control-plane,master 4m7s v1.22.5 +kub-3 Ready 3m7s v1.22.5 +``` + +Another nice test is the following: + +```ShellSession +kubectl get pods --all-namespaces -o wide +``` + +Which should yield something like the following: + +```ShellSession +$ kubectl get pods --all-namespaces -o wide +NAMESPACE NAME READY STATUS RESTARTS AGE IP NODE NOMINATED NODE READINESS GATES +kube-system coredns-8474476ff8-m2469 1/1 Running 0 2m45s 10.233.65.2 kub-2 +kube-system coredns-8474476ff8-v5wzj 1/1 Running 0 2m41s 10.233.64.3 kub-1 +kube-system dns-autoscaler-5ffdc7f89d-76tnv 1/1 Running 0 2m43s 10.233.64.2 kub-1 +kube-system kube-apiserver-kub-1 1/1 Running 1 4m54s 10.0.20.101 kub-1 +kube-system kube-apiserver-kub-2 1/1 Running 1 4m33s 10.0.20.102 kub-2 +kube-system kube-controller-manager-kub-1 1/1 Running 1 5m1s 10.0.20.101 kub-1 +kube-system kube-controller-manager-kub-2 1/1 Running 1 4m33s 10.0.20.102 kub-2 +kube-system kube-flannel-9xgf5 1/1 Running 0 3m10s 10.0.20.102 kub-2 +kube-system kube-flannel-l8jbl 1/1 Running 0 3m10s 10.0.20.101 kub-1 +kube-system kube-flannel-zss4t 1/1 Running 0 3m10s 10.0.20.103 kub-3 +kube-system kube-multus-ds-amd64-bhpc9 1/1 Running 0 3m2s 10.0.20.103 kub-3 +kube-system kube-multus-ds-amd64-n6vl8 1/1 Running 0 3m2s 10.0.20.102 kub-2 +kube-system kube-multus-ds-amd64-qttgs 1/1 Running 0 3m2s 10.0.20.101 kub-1 +kube-system kube-proxy-2x4jl 1/1 Running 0 3m33s 10.0.20.101 kub-1 +kube-system kube-proxy-d48r7 1/1 Running 0 3m33s 10.0.20.103 kub-3 +kube-system kube-proxy-f45lp 1/1 Running 0 3m33s 10.0.20.102 kub-2 +kube-system kube-scheduler-kub-1 1/1 Running 1 4m54s 10.0.20.101 kub-1 +kube-system kube-scheduler-kub-2 1/1 Running 1 4m33s 10.0.20.102 kub-2 +kube-system nginx-proxy-kub-3 1/1 Running 0 3m33s 10.0.20.103 kub-3 +kube-system nodelocaldns-cg9tz 1/1 Running 0 2m41s 10.0.20.102 kub-2 +kube-system nodelocaldns-htswt 1/1 Running 0 2m41s 10.0.20.103 kub-3 +kube-system nodelocaldns-nsp7s 1/1 Running 0 2m41s 10.0.20.101 kub-1 +local-path-storage local-path-provisioner-66df45bfdd-km4zg 1/1 Running 0 2m54s 10.233.66.2 kub-3 +``` diff --git a/kubespray/docs/vars.md b/kubespray/docs/vars.md new file mode 100644 index 0000000..3431d51 --- /dev/null +++ b/kubespray/docs/vars.md @@ -0,0 +1,322 @@ +# Configurable Parameters in Kubespray + +## Generic Ansible variables + +You can view facts gathered by Ansible automatically +[here](https://docs.ansible.com/ansible/latest/user_guide/playbooks_vars_facts.html#ansible-facts). + +Some variables of note include: + +* *ansible_user*: user to connect to via SSH +* *ansible_default_ipv4.address*: IP address Ansible automatically chooses. + Generated based on the output from the command ``ip -4 route get 8.8.8.8`` + +## Common vars that are used in Kubespray + +* *calico_version* - Specify version of Calico to use +* *calico_cni_version* - Specify version of Calico CNI plugin to use +* *docker_version* - Specify version of Docker to use (should be quoted + string). Must match one of the keys defined for *docker_versioned_pkg* + in `roles/container-engine/docker/vars/*.yml`. +* *containerd_version* - Specify version of containerd to use when setting `container_manager` to `containerd` +* *docker_containerd_version* - Specify which version of containerd to use when setting `container_manager` to `docker` +* *etcd_version* - Specify version of ETCD to use +* *calico_ipip_mode* - Configures Calico ipip encapsulation - valid values are 'Never', 'Always' and 'CrossSubnet' (default 'Never') +* *calico_vxlan_mode* - Configures Calico vxlan encapsulation - valid values are 'Never', 'Always' and 'CrossSubnet' (default 'Always') +* *calico_network_backend* - Configures Calico network backend - valid values are 'none', 'bird' and 'vxlan' (default 'vxlan') +* *kube_network_plugin* - Sets k8s network plugin (default Calico) +* *kube_proxy_mode* - Changes k8s proxy mode to iptables mode +* *kube_version* - Specify a given Kubernetes version +* *searchdomains* - Array of DNS domains to search when looking up hostnames +* *remove_default_searchdomains* - Boolean that removes the default searchdomain +* *nameservers* - Array of nameservers to use for DNS lookup +* *preinstall_selinux_state* - Set selinux state, permitted values are permissive, enforcing and disabled. + +## Addressing variables + +* *ip* - IP to use for binding services (host var) +* *access_ip* - IP for other hosts to use to connect to. Often required when + deploying from a cloud, such as OpenStack or GCE and you have separate + public/floating and private IPs. +* *ansible_default_ipv4.address* - Not Kubespray-specific, but it is used if ip + and access_ip are undefined +* *ip6* - IPv6 address to use for binding services. (host var) + If *enable_dual_stack_networks* is set to ``true`` and *ip6* is defined, + kubelet's ``--node-ip`` and node's ``InternalIP`` will be the combination of *ip* and *ip6*. +* *loadbalancer_apiserver* - If defined, all hosts will connect to this + address instead of localhost for kube_control_planes and kube_control_plane[0] for + kube_nodes. See more details in the + [HA guide](/docs/ha-mode.md). +* *loadbalancer_apiserver_localhost* - makes all hosts to connect to + the apiserver internally load balanced endpoint. Mutual exclusive to the + `loadbalancer_apiserver`. See more details in the + [HA guide](/docs/ha-mode.md). + +## Cluster variables + +Kubernetes needs some parameters in order to get deployed. These are the +following default cluster parameters: + +* *cluster_name* - Name of cluster (default is cluster.local) + +* *container_manager* - Container Runtime to install in the nodes (default is containerd) + +* *image_command_tool* - Tool used to pull images (default depends on `container_manager` + and is `nerdctl` for `containerd`, `crictl` for `crio`, `docker` for `docker`) + +* *image_command_tool_on_localhost* - Tool used to pull images on localhost + (default is equal to `image_command_tool`) + +* *dns_domain* - Name of cluster DNS domain (default is cluster.local) + +* *kube_network_plugin* - Plugin to use for container networking + +* *kube_service_addresses* - Subnet for cluster IPs (default is + 10.233.0.0/18). Must not overlap with kube_pods_subnet + +* *kube_pods_subnet* - Subnet for Pod IPs (default is 10.233.64.0/18). Must not + overlap with kube_service_addresses. + +* *kube_network_node_prefix* - Subnet allocated per-node for pod IPs. Remaining + bits in kube_pods_subnet dictates how many kube_nodes can be in cluster. Setting this > 25 will + raise an assertion in playbooks if the `kubelet_max_pods` var also isn't adjusted accordingly + (assertion not applicable to calico which doesn't use this as a hard limit, see + [Calico IP block sizes](https://docs.projectcalico.org/reference/resources/ippool#block-sizes)). + +* *enable_dual_stack_networks* - Setting this to true will provision both IPv4 and IPv6 networking for pods and services. + +* *kube_service_addresses_ipv6* - Subnet for cluster IPv6 IPs (default is ``fd85:ee78:d8a6:8607::1000/116``). Must not overlap with ``kube_pods_subnet_ipv6``. + +* *kube_pods_subnet_ipv6* - Subnet for Pod IPv6 IPs (default is ``fd85:ee78:d8a6:8607::1:0000/112``). Must not overlap with ``kube_service_addresses_ipv6``. + +* *kube_network_node_prefix_ipv6* - Subnet allocated per-node for pod IPv6 IPs. Remaining bits in ``kube_pods_subnet_ipv6`` dictates how many kube_nodes can be in cluster. + +* *skydns_server* - Cluster IP for DNS (default is 10.233.0.3) + +* *skydns_server_secondary* - Secondary Cluster IP for CoreDNS used with coredns_dual deployment (default is 10.233.0.4) + +* *enable_coredns_k8s_external* - If enabled, it configures the [k8s_external plugin](https://coredns.io/plugins/k8s_external/) + on the CoreDNS service. + +* *coredns_k8s_external_zone* - Zone that will be used when CoreDNS k8s_external plugin is enabled + (default is k8s_external.local) + +* *enable_coredns_k8s_endpoint_pod_names* - If enabled, it configures endpoint_pod_names option for kubernetes plugin. + on the CoreDNS service. + +* *cloud_provider* - Enable extra Kubelet option if operating inside GCE or + OpenStack (default is unset) + +* *kube_feature_gates* - A list of key=value pairs that describe feature gates for + alpha/experimental Kubernetes features. (defaults is `[]`). + Additionally, you can use also the following variables to individually customize your kubernetes components installation (they works exactly like `kube_feature_gates`): + * *kube_apiserver_feature_gates* + * *kube_controller_feature_gates* + * *kube_scheduler_feature_gates* + * *kube_proxy_feature_gates* + * *kubelet_feature_gates* + +* *kubeadm_feature_gates* - A list of key=value pairs that describe feature gates for + alpha/experimental Kubeadm features. (defaults is `[]`) + +* *authorization_modes* - A list of [authorization mode]( + https://kubernetes.io/docs/reference/access-authn-authz/authorization/#using-flags-for-your-authorization-module) + that the cluster should be configured for. Defaults to `['Node', 'RBAC']` + (Node and RBAC authorizers). + Note: `Node` and `RBAC` are enabled by default. Previously deployed clusters can be + converted to RBAC mode. However, your apps which rely on Kubernetes API will + require a service account and cluster role bindings. You can override this + setting by setting authorization_modes to `[]`. + +* *kube_apiserver_admission_control_config_file* - Enable configuration for `kube-apiserver` admission plugins. + Currently this variable allow you to configure the `EventRateLimit` admission plugin. + + To configure the **EventRateLimit** plugin you have to define a data structure like this: + +```yml +kube_apiserver_admission_event_rate_limits: + limit_1: + type: Namespace + qps: 50 + burst: 100 + cache_size: 2000 + limit_2: + type: User + qps: 50 + burst: 100 + ... +``` + +* *kube_apiserver_service_account_lookup* - Enable validation service account before validating token. Default `true`. + +Note, if cloud providers have any use of the ``10.233.0.0/16``, like instances' +private addresses, make sure to pick another values for ``kube_service_addresses`` +and ``kube_pods_subnet``, for example from the ``172.18.0.0/16``. + +## Enabling Dual Stack (IPV4 + IPV6) networking + +If *enable_dual_stack_networks* is set to ``true``, Dual Stack networking will be enabled in the cluster. This will use the default IPv4 and IPv6 subnets specified in the defaults file in the ``kubespray-defaults`` role, unless overridden of course. The default config will give you room for up to 256 nodes with 126 pods per node, and up to 4096 services. + +## DNS variables + +By default, hosts are set up with 8.8.8.8 as an upstream DNS server and all +other settings from your existing /etc/resolv.conf are lost. Set the following +variables to match your requirements. + +* *upstream_dns_servers* - Array of upstream DNS servers configured on host in + addition to Kubespray deployed DNS +* *nameservers* - Array of DNS servers configured for use by hosts +* *searchdomains* - Array of up to 4 search domains +* *remove_default_searchdomains* - Boolean. If enabled, `searchdomains` variable can hold 6 search domains. +* *dns_etchosts* - Content of hosts file for coredns and nodelocaldns +* *dns_upstream_forward_extra_opts* - Options to add in the forward section of coredns/nodelocaldns related to upstream DNS servers + +For more information, see [DNS +Stack](https://github.com/kubernetes-sigs/kubespray/blob/master/docs/dns-stack.md). + +## Other service variables + +* *docker_options* - Commonly used to set + ``--insecure-registry=myregistry.mydomain:5000`` + +* *docker_plugins* - This list can be used to define [Docker plugins](https://docs.docker.com/engine/extend/) to install. + +* *containerd_default_runtime* - If defined, changes the default Containerd runtime used by the Kubernetes CRI plugin. + +* *containerd_additional_runtimes* - Sets the additional Containerd runtimes used by the Kubernetes CRI plugin. + [Default config](https://github.com/kubernetes-sigs/kubespray/blob/master/roles/container-engine/containerd/defaults/main.yml) can be overridden in inventory vars. + +* *http_proxy/https_proxy/no_proxy/no_proxy_exclude_workers/additional_no_proxy* - Proxy variables for deploying behind a + proxy. Note that no_proxy defaults to all internal cluster IPs and hostnames + that correspond to each node. + +* *kubelet_cgroup_driver* - Allows manual override of the cgroup-driver option for Kubelet. + By default autodetection is used to match container manager configuration. + `systemd` is the preferred driver for `containerd` though it can have issues with `cgroups v1` and `kata-containers` in which case you may want to change to `cgroupfs`. + +* *kubelet_rotate_certificates* - Auto rotate the kubelet client certificates by requesting new certificates + from the kube-apiserver when the certificate expiration approaches. + +* *kubelet_rotate_server_certificates* - Auto rotate the kubelet server certificates by requesting new certificates + from the kube-apiserver when the certificate expiration approaches. + Note that enabling this also activates *kubelet_csr_approver* which approves automatically the CSRs. + To customize its behavior, you can override the Helm values via *kubelet_csr_approver_values*. + See [kubelet-csr-approver](https://github.com/postfinance/kubelet-csr-approver) for more information. + +* *kubelet_streaming_connection_idle_timeout* - Set the maximum time a streaming connection can be idle before the connection is automatically closed. + +* *kubelet_image_gc_high_threshold* - Set the percent of disk usage after which image garbage collection is always run. + The percent is calculated by dividing this field value by 100, so this field must be between 0 and 100, inclusive. + When specified, the value must be greater than imageGCLowThresholdPercent. Default: 85 + +* *kubelet_image_gc_low_threshold* - Set the percent of disk usage before which image garbage collection is never run. + Lowest disk usage to garbage collect to. + The percent is calculated by dividing this field value by 100, so the field value must be between 0 and 100, inclusive. + When specified, the value must be less than imageGCHighThresholdPercent. Default: 80 + +* *kubelet_make_iptables_util_chains* - If `true`, causes the kubelet ensures a set of `iptables` rules are present on host. + +* *kubelet_cpu_manager_policy* - If set to `static`, allows pods with certain resource characteristics to be granted increased CPU affinity and exclusivity on the node. And it should be set with `kube_reserved` or `system-reserved`, enable this with the following guide:[Control CPU Management Policies on the Node](https://kubernetes.io/docs/tasks/administer-cluster/cpu-management-policies/) + +* *kubelet_topology_manager_policy* - Control the behavior of the allocation of CPU and Memory from different [NUMA](https://en.wikipedia.org/wiki/Non-uniform_memory_access) Nodes. Enable this with the following guide: [Control Topology Management Policies on a node](https://kubernetes.io/docs/tasks/administer-cluster/topology-manager). + +* *kubelet_topology_manager_scope* - The Topology Manager can deal with the alignment of resources in a couple of distinct scopes: `container` and `pod`. See [Topology Manager Scopes](https://kubernetes.io/docs/tasks/administer-cluster/topology-manager/#topology-manager-scopes). + +* *kubelet_systemd_hardening* - If `true`, provides kubelet systemd service with security features for isolation. + + **N.B.** To enable this feature, ensure you are using the **`cgroup v2`** on your system. Check it out with command: `sudo ls -l /sys/fs/cgroup/*.slice`. If directory does not exist, enable this with the following guide: [enable cgroup v2](https://rootlesscontaine.rs/getting-started/common/cgroup2/#enabling-cgroup-v2). + + * *kubelet_secure_addresses* - By default *kubelet_systemd_hardening* set the **control plane** `ansible_host` IPs as the `kubelet_secure_addresses`. In case you have multiple interfaces in your control plane nodes and the `kube-apiserver` is not bound to the default interface, you can override them with this variable. + Example: + + The **control plane** node may have 2 interfaces with the following IP addresses: `eth0:10.0.0.110`, `eth1:192.168.1.110`. + + By default the `kubelet_secure_addresses` is set with the `10.0.0.110` the ansible control host uses `eth0` to connect to the machine. In case you want to use `eth1` as the outgoing interface on which `kube-apiserver` connects to the `kubelet`s, you should override the variable in this way: `kubelet_secure_addresses: "192.168.1.110"`. + +* *node_labels* - Labels applied to nodes via `kubectl label node`. + For example, labels can be set in the inventory as variables or more widely in group_vars. + *node_labels* can only be defined as a dict: + +```yml +node_labels: + label1_name: label1_value + label2_name: label2_value +``` + +* *node_taints* - Taints applied to nodes via kubelet --register-with-taints parameter. + For example, taints can be set in the inventory as variables or more widely in group_vars. + *node_taints* has to be defined as a list of strings in format `key=value:effect`, e.g.: + +```yml +node_taints: + - "node.example.com/external=true:NoSchedule" +``` + +* *podsecuritypolicy_enabled* - When set to `true`, enables the PodSecurityPolicy admission controller and defines two policies `privileged` (applying to all resources in `kube-system` namespace and kubelet) and `restricted` (applying all other namespaces). + Addons deployed in kube-system namespaces are handled. +* *kubernetes_audit* - When set to `true`, enables Auditing. + The auditing parameters can be tuned via the following variables (which default values are shown below): + * `audit_log_path`: /var/log/audit/kube-apiserver-audit.log + * `audit_log_maxage`: 30 + * `audit_log_maxbackups`: 10 + * `audit_log_maxsize`: 100 + * `audit_policy_file`: "{{ kube_config_dir }}/audit-policy/apiserver-audit-policy.yaml" + + By default, the `audit_policy_file` contains [default rules](https://github.com/kubernetes-sigs/kubespray/blob/master/roles/kubernetes/control-plane/templates/apiserver-audit-policy.yaml.j2) that can be overridden with the `audit_policy_custom_rules` variable. +* *kubernetes_audit_webhook* - When set to `true`, enables the webhook audit backend. + The webhook parameters can be tuned via the following variables (which default values are shown below): + * `audit_webhook_config_file`: "{{ kube_config_dir }}/audit-policy/apiserver-audit-webhook-config.yaml" + * `audit_webhook_server_url`: `"https://audit.app"` + * `audit_webhook_server_extra_args`: {} + * `audit_webhook_mode`: batch + * `audit_webhook_batch_max_size`: 100 + * `audit_webhook_batch_max_wait`: 1s + +### Custom flags for Kube Components + +For all kube components, custom flags can be passed in. This allows for edge cases where users need changes to the default deployment that may not be applicable to all deployments. + +Extra flags for the kubelet can be specified using these variables, +in the form of dicts of key-value pairs of configuration parameters that will be inserted into the kubelet YAML config file. The `kubelet_node_config_extra_args` apply kubelet settings only to nodes and not control planes. Example: + +```yml +kubelet_config_extra_args: + evictionHard: + memory.available: "100Mi" + evictionSoftGracePeriod: + memory.available: "30s" + evictionSoft: + memory.available: "300Mi" +``` + +The possible vars are: + +* *kubelet_config_extra_args* +* *kubelet_node_config_extra_args* + +Previously, the same parameters could be passed as flags to kubelet binary with the following vars: + +* *kubelet_custom_flags* +* *kubelet_node_custom_flags* + +The `kubelet_node_custom_flags` apply kubelet settings only to nodes and not control planes. Example: + +```yml +kubelet_custom_flags: + - "--eviction-hard=memory.available<100Mi" + - "--eviction-soft-grace-period=memory.available=30s" + - "--eviction-soft=memory.available<300Mi" +``` + +This alternative is deprecated and will remain until the flags are completely removed from kubelet + +Extra flags for the API server, controller, and scheduler components can be specified using these variables, +in the form of dicts of key-value pairs of configuration parameters that will be inserted into the kubeadm YAML config file: + +* *kube_kubeadm_apiserver_extra_args* +* *kube_kubeadm_controller_extra_args* +* *kube_kubeadm_scheduler_extra_args* + +## App variables + +* *helm_version* - Only supports v3.x. Existing v2 installs (with Tiller) will not be modified and need to be removed manually. diff --git a/kubespray/docs/vsphere-csi.md b/kubespray/docs/vsphere-csi.md new file mode 100644 index 0000000..af58440 --- /dev/null +++ b/kubespray/docs/vsphere-csi.md @@ -0,0 +1,102 @@ +# vSphere CSI Driver + +vSphere CSI driver allows you to provision volumes over a vSphere deployment. The Kubernetes historic in-tree cloud provider is deprecated and will be removed in future versions. + +## Prerequisites + +The vSphere user for CSI driver requires a set of privileges to perform Cloud Native Storage operations. Follow the [official guide](https://vsphere-csi-driver.sigs.k8s.io/driver-deployment/prerequisites.html#roles_and_privileges) to configure those. + +## Kubespray configuration + +To enable vSphere CSI driver, uncomment the `vsphere_csi_enabled` option in `group_vars/all/vsphere.yml` and set it to `true`. + +To set the number of replicas for the vSphere CSI controller, you can change `vsphere_csi_controller_replicas` option in `group_vars/all/vsphere.yml`. + +You need to source the vSphere credentials you use to deploy your machines that will host Kubernetes. + +| Variable | Required | Type | Choices | Default | Comment | +|-------------------------------------------------|----------|---------|-----------------|-------------------------|-----------------------------------------------------------------------------------------------------------------------------| +| external_vsphere_vcenter_ip | TRUE | string | | | IP/URL of the vCenter | +| external_vsphere_vcenter_port | TRUE | string | | "443" | Port of the vCenter API | +| external_vsphere_insecure | TRUE | string | "true", "false" | "true" | set to "true" if the host above uses a self-signed cert | +| external_vsphere_user | TRUE | string | | | User name for vCenter with required privileges (Can also be specified with the `VSPHERE_USER` environment variable) | +| external_vsphere_password | TRUE | string | | | Password for vCenter (Can also be specified with the `VSPHERE_PASSWORD` environment variable) | +| external_vsphere_datacenter | TRUE | string | | | Datacenter name to use | +| external_vsphere_kubernetes_cluster_id | TRUE | string | | "kubernetes-cluster-id" | Kubernetes cluster ID to use | +| external_vsphere_version | TRUE | string | | "6.7u3" | Vmware Vsphere version where located all VMs | +| external_vsphere_cloud_controller_image_tag | TRUE | string | | "latest" | Kubernetes cluster ID to use | +| vsphere_syncer_image_tag | TRUE | string | | "v2.2.1" | Syncer image tag to use | +| vsphere_csi_attacher_image_tag | TRUE | string | | "v3.1.0" | CSI attacher image tag to use | +| vsphere_csi_controller | TRUE | string | | "v2.2.1" | CSI controller image tag to use | +| vsphere_csi_controller_replicas | TRUE | integer | | 1 | Number of pods Kubernetes should deploy for the CSI controller | +| vsphere_csi_liveness_probe_image_tag | TRUE | string | | "v2.2.0" | CSI liveness probe image tag to use | +| vsphere_csi_provisioner_image_tag | TRUE | string | | "v2.1.0" | CSI provisioner image tag to use | +| vsphere_csi_node_driver_registrar_image_tag | TRUE | string | | "v1.1.0" | CSI node driver registrar image tag to use | +| vsphere_csi_driver_image_tag | TRUE | string | | "v1.0.2" | CSI driver image tag to use | +| vsphere_csi_resizer_tag | TRUE | string | | "v1.1.0" | CSI resizer image tag to use | +| vsphere_csi_aggressive_node_drain | FALSE | boolean | | false | Enable aggressive node drain strategy | +| vsphere_csi_aggressive_node_unreachable_timeout | FALSE | int | 300 | | Timeout till node will be drained when it in an unreachable state | +| vsphere_csi_aggressive_node_not_ready_timeout | FALSE | int | 300 | | Timeout till node will be drained when it in not-ready state | +| vsphere_csi_namespace | TRUE | string | | "kube-system" | vSphere CSI namespace to use; kube-system for backward compatibility, should be change to vmware-system-csi on the long run | + +## Usage example + +To test the dynamic provisioning using vSphere CSI driver, make sure to create a [storage policy](https://github.com/kubernetes/cloud-provider-vsphere/blob/master/docs/book/tutorials/kubernetes-on-vsphere-with-kubeadm.md#create-a-storage-policy) and [storage class](https://github.com/kubernetes/cloud-provider-vsphere/blob/master/docs/book/tutorials/kubernetes-on-vsphere-with-kubeadm.md#create-a-storageclass), then apply the following manifest: + +```yml +--- +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + name: csi-pvc-vsphere +spec: + accessModes: + - ReadWriteOnce + resources: + requests: + storage: 1Gi + storageClassName: mongodb-sc + +--- +apiVersion: v1 +kind: Pod +metadata: + name: nginx +spec: + containers: + - image: nginx + imagePullPolicy: IfNotPresent + name: nginx + ports: + - containerPort: 80 + protocol: TCP + volumeMounts: + - mountPath: /usr/share/nginx/html + name: csi-data-vsphere + volumes: + - name: csi-data-vsphere + persistentVolumeClaim: + claimName: csi-pvc-vsphere + readOnly: false +``` + +Apply this conf to your cluster: ```kubectl apply -f nginx.yml``` + +You should see the PVC provisioned and bound: + +```ShellSession +$ kubectl get pvc +NAME STATUS VOLUME CAPACITY ACCESS MODES STORAGECLASS AGE +csi-pvc-vsphere Bound pvc-dc7b1d21-ee41-45e1-98d9-e877cc1533ac 1Gi RWO mongodb-sc 10s +``` + +And the volume mounted to the Nginx Pod (wait until the Pod is Running): + +```ShellSession +kubectl exec -it nginx -- df -h | grep /usr/share/nginx/html +/dev/sdb 976M 2.6M 907M 1% /usr/share/nginx/html +``` + +## More info + +For further information about the vSphere CSI Driver, you can refer to the official [vSphere Cloud Provider documentation](https://cloud-provider-vsphere.sigs.k8s.io/container_storage_interface.html). diff --git a/kubespray/docs/vsphere.md b/kubespray/docs/vsphere.md new file mode 100644 index 0000000..a75a25d --- /dev/null +++ b/kubespray/docs/vsphere.md @@ -0,0 +1,134 @@ +# vSphere + +Kubespray can be deployed with vSphere as Cloud provider. This feature supports: + +- Volumes +- Persistent Volumes +- Storage Classes and provisioning of volumes +- vSphere Storage Policy Based Management for Containers orchestrated by Kubernetes + +## Out-of-tree vSphere cloud provider + +### Prerequisites + +You need at first to configure your vSphere environment by following the [official documentation](https://github.com/kubernetes/cloud-provider-vsphere/blob/master/docs/book/tutorials/kubernetes-on-vsphere-with-kubeadm.md#prerequisites). + +After this step you should have: + +- vSphere upgraded to 6.7 U3 or later +- VM hardware upgraded to version 15 or higher +- UUID activated for each VM where Kubernetes will be deployed + +### Kubespray configuration + +First in `inventory/sample/group_vars/all/all.yml` you must set the `cloud_provider` to `external` and `external_cloud_provider` to `vsphere`. + +```yml +cloud_provider: "external" +external_cloud_provider: "vsphere" +``` + +Then, `inventory/sample/group_vars/all/vsphere.yml`, you need to declare your vCenter credentials and enable the vSphere CSI following the description below. + +| Variable | Required | Type | Choices | Default | Comment | +|----------------------------------------|----------|---------|----------------------------|---------------------------|---------------------------------------------------------------------------------------------------------------------| +| external_vsphere_vcenter_ip | TRUE | string | | | IP/URL of the vCenter | +| external_vsphere_vcenter_port | TRUE | string | | "443" | Port of the vCenter API | +| external_vsphere_insecure | TRUE | string | "true", "false" | "true" | set to "true" if the host above uses a self-signed cert | +| external_vsphere_user | TRUE | string | | | User name for vCenter with required privileges (Can also be specified with the `VSPHERE_USER` environment variable) | +| external_vsphere_password | TRUE | string | | | Password for vCenter (Can also be specified with the `VSPHERE_PASSWORD` environment variable) | +| external_vsphere_datacenter | TRUE | string | | | Datacenter name to use | +| external_vsphere_kubernetes_cluster_id | TRUE | string | | "kubernetes-cluster-id" | Kubernetes cluster ID to use | +| vsphere_csi_enabled | TRUE | boolean | | false | Enable vSphere CSI | + +Example configuration: + +```yml +external_vsphere_vcenter_ip: "myvcenter.domain.com" +external_vsphere_vcenter_port: "443" +external_vsphere_insecure: "true" +external_vsphere_user: "administrator@vsphere.local" +external_vsphere_password: "K8s_admin" +external_vsphere_datacenter: "DATACENTER_name" +external_vsphere_kubernetes_cluster_id: "kubernetes-cluster-id" +vsphere_csi_enabled: true +``` + +For a more fine-grained CSI setup, refer to the [vsphere-csi](/docs/vsphere-csi.md) documentation. + +### Deployment + +Once the configuration is set, you can execute the playbook again to apply the new configuration: + +```ShellSession +cd kubespray +ansible-playbook -i inventory/sample/hosts.ini -b -v cluster.yml +``` + +You'll find some useful examples [here](https://github.com/kubernetes/cloud-provider-vsphere/blob/master/docs/book/tutorials/kubernetes-on-vsphere-with-kubeadm.md#sample-manifests-to-test-csi-driver-functionality) to test your configuration. + +## In-tree vSphere cloud provider ([deprecated](https://cloud-provider-vsphere.sigs.k8s.io/concepts/in_tree_vs_out_of_tree.html)) + +### Prerequisites (deprecated) + +You need at first to configure your vSphere environment by following the [official documentation](https://kubernetes.io/docs/getting-started-guides/vsphere/#vsphere-cloud-provider). + +After this step you should have: + +- UUID activated for each VM where Kubernetes will be deployed +- A vSphere account with required privileges + +If you intend to leverage the [zone and region node labeling](https://kubernetes.io/docs/reference/kubernetes-api/labels-annotations-taints/#failure-domain-beta-kubernetes-io-region), create a tag category for both the zone and region in vCenter. The tags can then be applied at the host, cluster, datacenter, or folder level, and the cloud provider will walk the hierarchy to extract and apply the labels to the Kubernetes nodes. + +### Kubespray configuration (deprecated) + +First you must define the cloud provider in `inventory/sample/group_vars/all.yml` and set it to `vsphere`. + +```yml +cloud_provider: vsphere +``` + +Then, in the same file, you need to declare your vCenter credentials following the description below. + +| Variable | Required | Type | Choices | Default | Comment | +|------------------------------|----------|---------|----------------------------|---------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| vsphere_vcenter_ip | TRUE | string | | | IP/URL of the vCenter | +| vsphere_vcenter_port | TRUE | integer | | | Port of the vCenter API. Commonly 443 | +| vsphere_insecure | TRUE | integer | 1, 0 | | set to 1 if the host above uses a self-signed cert | +| vsphere_user | TRUE | string | | | User name for vCenter with required privileges | +| vsphere_password | TRUE | string | | | Password for vCenter | +| vsphere_datacenter | TRUE | string | | | Datacenter name to use | +| vsphere_datastore | TRUE | string | | | Datastore name to use | +| vsphere_working_dir | TRUE | string | | | Working directory from the view "VMs and template" in the vCenter where VM are placed | +| vsphere_scsi_controller_type | TRUE | string | buslogic, pvscsi, parallel | pvscsi | SCSI controller name. Commonly "pvscsi". | +| vsphere_vm_uuid | FALSE | string | | | VM Instance UUID of virtual machine that host K8s master. Can be retrieved from instanceUuid property in VmConfigInfo, or as vc.uuid in VMX file or in `/sys/class/dmi/id/product_serial` (Optional, only used for Kubernetes <= 1.9.2) | +| vsphere_public_network | FALSE | string | | Blank | Name of the network the VMs are joined to | +| vsphere_resource_pool | FALSE | string | | Blank | Name of the Resource pool where the VMs are located (Optional, only used for Kubernetes >= 1.9.2) | +| vsphere_zone_category | FALSE | string | | | Name of the tag category used to set the `failure-domain.beta.kubernetes.io/zone` label on nodes (Optional, only used for Kubernetes >= 1.12.0) | +| vsphere_region_category | FALSE | string | | | Name of the tag category used to set the `failure-domain.beta.kubernetes.io/region` label on nodes (Optional, only used for Kubernetes >= 1.12.0) | + +Example configuration: + +```yml +vsphere_vcenter_ip: "myvcenter.domain.com" +vsphere_vcenter_port: 443 +vsphere_insecure: 1 +vsphere_user: "k8s@vsphere.local" +vsphere_password: "K8s_admin" +vsphere_datacenter: "DATACENTER_name" +vsphere_datastore: "DATASTORE_name" +vsphere_working_dir: "Docker_hosts" +vsphere_scsi_controller_type: "pvscsi" +vsphere_resource_pool: "K8s-Pool" +``` + +### Deployment (deprecated) + +Once the configuration is set, you can execute the playbook again to apply the new configuration: + +```ShellSession +cd kubespray +ansible-playbook -i inventory/sample/hosts.ini -b -v cluster.yml +``` + +You'll find some useful examples [here](https://github.com/kubernetes/examples/tree/master/staging/volumes/vsphere) to test your configuration. diff --git a/kubespray/docs/weave.md b/kubespray/docs/weave.md new file mode 100644 index 0000000..30fa494 --- /dev/null +++ b/kubespray/docs/weave.md @@ -0,0 +1,79 @@ +# Weave + +Weave 2.0.1 is supported by kubespray + +Weave uses [**consensus**](https://www.weave.works/docs/net/latest/ipam/##consensus) mode (default mode) and [**seed**](https://www.weave.works/docs/net/latest/ipam/#seed) mode. + +`Consensus` mode is best to use on static size cluster and `seed` mode is best to use on dynamic size cluster + +Weave encryption is supported for all communication + +* To use Weave encryption, specify a strong password (if no password, no encryption) + +```ShellSession +# In file ./inventory/sample/group_vars/k8s_cluster.yml +weave_password: EnterPasswordHere +``` + +This password is used to set an environment variable inside weave container. + +Weave is deployed by kubespray using a daemonSet + +* Check the status of Weave containers + +```ShellSession +# From client +kubectl -n kube-system get pods | grep weave +# output +weave-net-50wd2 2/2 Running 0 2m +weave-net-js9rb 2/2 Running 0 2m +``` + +There must be as many pods as nodes (here kubernetes have 2 nodes so there are 2 weave pods). + +* Check status of weave (connection,encryption ...) for each node + +```ShellSession +# On nodes +curl http://127.0.0.1:6784/status +# output on node1 +Version: 2.0.1 (up to date; next check at 2017/08/01 13:51:34) + + Service: router + Protocol: weave 1..2 + Name: fa:16:3e:b3:d6:b2(node1) + Encryption: enabled + PeerDiscovery: enabled + Targets: 2 + Connections: 2 (1 established, 1 failed) + Peers: 2 (with 2 established connections) + TrustedSubnets: none + + Service: ipam + Status: ready + Range: 10.233.64.0/18 + DefaultSubnet: 10.233.64.0/18 +``` + +* Check parameters of weave for each node + +```ShellSession +# On nodes +ps -aux | grep weaver +# output on node1 (here its use seed mode) +root 8559 0.2 3.0 365280 62700 ? Sl 08:25 0:00 /home/weave/weaver --name=fa:16:3e:b3:d6:b2 --port=6783 --datapath=datapath --host-root=/host --http-addr=127.0.0.1:6784 --status-addr=0.0.0.0:6782 --docker-api= --no-dns --db-prefix=/weavedb/weave-net --ipalloc-range=10.233.64.0/18 --nickname=node1 --ipalloc-init seed=fa:16:3e:b3:d6:b2,fa:16:3e:f0:50:53 --conn-limit=30 --expect-npc 192.168.208.28 192.168.208.19 +``` + +## Consensus mode (default mode) + +This mode is best to use on static size cluster + +### Seed mode + +This mode is best to use on dynamic size cluster + +The seed mode also allows multi-clouds and hybrid on-premise/cloud clusters deployment. + +* Switch from consensus mode to seed/Observation mode + +See [weave ipam documentation](https://www.weave.works/docs/net/latest/tasks/ipam/ipam/) and use `weave_extra_args` to enable. diff --git a/kubespray/extra_playbooks/files/get_cinder_pvs.sh b/kubespray/extra_playbooks/files/get_cinder_pvs.sh new file mode 100644 index 0000000..73a088e --- /dev/null +++ b/kubespray/extra_playbooks/files/get_cinder_pvs.sh @@ -0,0 +1,2 @@ +#!/bin/sh +kubectl get pv -o go-template --template='{{ range .items }}{{ $metadata := .metadata }}{{ with $value := index .metadata.annotations "pv.kubernetes.io/provisioned-by" }}{{ if eq $value "kubernetes.io/cinder" }}{{printf "%s\n" $metadata.name}}{{ end }}{{ end }}{{ end }}' diff --git a/kubespray/extra_playbooks/inventory b/kubespray/extra_playbooks/inventory new file mode 120000 index 0000000..e09e1ad --- /dev/null +++ b/kubespray/extra_playbooks/inventory @@ -0,0 +1 @@ +../inventory \ No newline at end of file diff --git a/kubespray/extra_playbooks/migrate_openstack_provider.yml b/kubespray/extra_playbooks/migrate_openstack_provider.yml new file mode 100644 index 0000000..a82a587 --- /dev/null +++ b/kubespray/extra_playbooks/migrate_openstack_provider.yml @@ -0,0 +1,30 @@ +--- +- name: Remove old cloud provider config + hosts: kube_node:kube_control_plane + tasks: + - name: Remove old cloud provider config + file: + path: "{{ item }}" + state: absent + with_items: + - /etc/kubernetes/cloud_config +- name: Migrate intree Cinder PV + hosts: kube_control_plane[0] + tasks: + - name: Include kubespray-default variables + include_vars: ../roles/kubespray-defaults/defaults/main.yaml + - name: Copy get_cinder_pvs.sh to master + copy: + src: get_cinder_pvs.sh + dest: /tmp + mode: u+rwx + - name: Get PVs provisioned by in-tree cloud provider + command: /tmp/get_cinder_pvs.sh + register: pvs + - name: Remove get_cinder_pvs.sh + file: + path: /tmp/get_cinder_pvs.sh + state: absent + - name: Rewrite the "pv.kubernetes.io/provisioned-by" annotation + command: "{{ bin_dir }}/kubectl annotate --overwrite pv {{ item }} pv.kubernetes.io/provisioned-by=cinder.csi.openstack.org" + loop: "{{ pvs.stdout_lines | list }}" diff --git a/kubespray/extra_playbooks/roles b/kubespray/extra_playbooks/roles new file mode 120000 index 0000000..d8c4472 --- /dev/null +++ b/kubespray/extra_playbooks/roles @@ -0,0 +1 @@ +../roles \ No newline at end of file diff --git a/kubespray/extra_playbooks/upgrade-only-k8s.yml b/kubespray/extra_playbooks/upgrade-only-k8s.yml new file mode 100644 index 0000000..4207f8d --- /dev/null +++ b/kubespray/extra_playbooks/upgrade-only-k8s.yml @@ -0,0 +1,61 @@ +--- +### NOTE: This playbook cannot be used to deploy any new nodes to the cluster. +### Additional information: +### * Will not upgrade etcd +### * Will not upgrade network plugins +### * Will not upgrade Docker +### * Will not pre-download containers or kubeadm +### * Currently does not support Vault deployment. +### +### In most cases, you probably want to use upgrade-cluster.yml playbook and +### not this one. + +- name: Setup ssh config to use the bastion + hosts: localhost + gather_facts: False + roles: + - { role: kubespray-defaults} + - { role: bastion-ssh-config, tags: ["localhost", "bastion"]} + +- name: Bootstrap hosts OS for Ansible + hosts: k8s_cluster:etcd:calico_rr + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + gather_facts: false + vars: + # Need to disable pipelining for bootstrap-os as some systems have requiretty in sudoers set, which makes pipelining + # fail. bootstrap-os fixes this on these systems, so in later plays it can be enabled. + ansible_ssh_pipelining: false + roles: + - { role: kubespray-defaults} + - { role: bootstrap-os, tags: bootstrap-os} + +- name: Preinstall + hosts: k8s_cluster:etcd:calico_rr + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + roles: + - { role: kubespray-defaults} + - { role: kubernetes/preinstall, tags: preinstall } + +- name: Handle upgrades to master components first to maintain backwards compat. + hosts: kube_control_plane + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + serial: 1 + roles: + - { role: kubespray-defaults} + - { role: upgrade/pre-upgrade, tags: pre-upgrade } + - { role: kubernetes/node, tags: node } + - { role: kubernetes/control-plane, tags: master, upgrade_cluster_setup: true } + - { role: kubernetes/client, tags: client } + - { role: kubernetes-apps/cluster_roles, tags: cluster-roles } + - { role: upgrade/post-upgrade, tags: post-upgrade } + +- name: Finally handle worker upgrades, based on given batch size + hosts: kube_node:!kube_control_plane + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + serial: "{{ serial | default('20%') }}" + roles: + - { role: kubespray-defaults} + - { role: upgrade/pre-upgrade, tags: pre-upgrade } + - { role: kubernetes/node, tags: node } + - { role: upgrade/post-upgrade, tags: post-upgrade } + - { role: kubespray-defaults} diff --git a/kubespray/extra_playbooks/wait-for-cloud-init.yml b/kubespray/extra_playbooks/wait-for-cloud-init.yml new file mode 100644 index 0000000..82c4194 --- /dev/null +++ b/kubespray/extra_playbooks/wait-for-cloud-init.yml @@ -0,0 +1,6 @@ +--- +- name: Wait for cloud-init to finish + hosts: all + tasks: + - name: Wait for cloud-init to finish + command: cloud-init status --wait diff --git a/kubespray/inventory/local/group_vars b/kubespray/inventory/local/group_vars new file mode 120000 index 0000000..a30ba68 --- /dev/null +++ b/kubespray/inventory/local/group_vars @@ -0,0 +1 @@ +../sample/group_vars \ No newline at end of file diff --git a/kubespray/inventory/local/hosts.ini b/kubespray/inventory/local/hosts.ini new file mode 100644 index 0000000..4a6197e --- /dev/null +++ b/kubespray/inventory/local/hosts.ini @@ -0,0 +1,14 @@ +node1 ansible_connection=local local_release_dir={{ansible_env.HOME}}/releases + +[kube_control_plane] +node1 + +[etcd] +node1 + +[kube_node] +node1 + +[k8s_cluster:children] +kube_node +kube_control_plane diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/all.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/all.yml new file mode 100644 index 0000000..b93f1a3 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/all.yml @@ -0,0 +1,139 @@ +--- +## Directory where the binaries will be installed +bin_dir: /usr/local/bin + +## The access_ip variable is used to define how other nodes should access +## the node. This is used in flannel to allow other flannel nodes to see +## this node for example. The access_ip is really useful AWS and Google +## environments where the nodes are accessed remotely by the "public" ip, +## but don't know about that address themselves. +# access_ip: 1.1.1.1 + + +## External LB example config +## apiserver_loadbalancer_domain_name: "elb.some.domain" +# loadbalancer_apiserver: +# address: 1.2.3.4 +# port: 1234 + +## Internal loadbalancers for apiservers +# loadbalancer_apiserver_localhost: true +# valid options are "nginx" or "haproxy" +# loadbalancer_apiserver_type: nginx # valid values "nginx" or "haproxy" + +## Local loadbalancer should use this port +## And must be set port 6443 +loadbalancer_apiserver_port: 6443 + +## If loadbalancer_apiserver_healthcheck_port variable defined, enables proxy liveness check for nginx. +loadbalancer_apiserver_healthcheck_port: 8081 + +### OTHER OPTIONAL VARIABLES + +## By default, Kubespray collects nameservers on the host. It then adds the previously collected nameservers in nameserverentries. +## If true, Kubespray does not include host nameservers in nameserverentries in dns_late stage. However, It uses the nameserver to make sure cluster installed safely in dns_early stage. +## Use this option with caution, you may need to define your dns servers. Otherwise, the outbound queries such as www.google.com may fail. +# disable_host_nameservers: false + +## Upstream dns servers +# upstream_dns_servers: +# - 8.8.8.8 +# - 8.8.4.4 + +## There are some changes specific to the cloud providers +## for instance we need to encapsulate packets with some network plugins +## If set the possible values are either 'gce', 'aws', 'azure', 'openstack', 'vsphere', 'oci', or 'external' +## When openstack is used make sure to source in the openstack credentials +## like you would do when using openstack-client before starting the playbook. +# cloud_provider: + +## When cloud_provider is set to 'external', you can set the cloud controller to deploy +## Supported cloud controllers are: 'openstack', 'vsphere', 'huaweicloud' and 'hcloud' +## When openstack or vsphere are used make sure to source in the required fields +# external_cloud_provider: + +## Set these proxy values in order to update package manager and docker daemon to use proxies and custom CA for https_proxy if needed +# http_proxy: "" +# https_proxy: "" +# https_proxy_cert_file: "" + +## Refer to roles/kubespray-defaults/defaults/main.yml before modifying no_proxy +# no_proxy: "" + +## Some problems may occur when downloading files over https proxy due to ansible bug +## https://github.com/ansible/ansible/issues/32750. Set this variable to False to disable +## SSL validation of get_url module. Note that kubespray will still be performing checksum validation. +# download_validate_certs: False + +## If you need exclude all cluster nodes from proxy and other resources, add other resources here. +# additional_no_proxy: "" + +## If you need to disable proxying of os package repositories but are still behind an http_proxy set +## skip_http_proxy_on_os_packages to true +## This will cause kubespray not to set proxy environment in /etc/yum.conf for centos and in /etc/apt/apt.conf for debian/ubuntu +## Special information for debian/ubuntu - you have to set the no_proxy variable, then apt package will install from your source of wish +# skip_http_proxy_on_os_packages: false + +## Since workers are included in the no_proxy variable by default, docker engine will be restarted on all nodes (all +## pods will restart) when adding or removing workers. To override this behaviour by only including master nodes in the +## no_proxy variable, set below to true: +no_proxy_exclude_workers: false + +## Certificate Management +## This setting determines whether certs are generated via scripts. +## Chose 'none' if you provide your own certificates. +## Option is "script", "none" +# cert_management: script + +## Set to true to allow pre-checks to fail and continue deployment +# ignore_assert_errors: false + +## The read-only port for the Kubelet to serve on with no authentication/authorization. Uncomment to enable. +# kube_read_only_port: 10255 + +## Set true to download and cache container +# download_container: true + +## Deploy container engine +# Set false if you want to deploy container engine manually. +# deploy_container_engine: true + +## Red Hat Enterprise Linux subscription registration +## Add either RHEL subscription Username/Password or Organization ID/Activation Key combination +## Update RHEL subscription purpose usage, role and SLA if necessary +# rh_subscription_username: "" +# rh_subscription_password: "" +# rh_subscription_org_id: "" +# rh_subscription_activation_key: "" +# rh_subscription_usage: "Development" +# rh_subscription_role: "Red Hat Enterprise Server" +# rh_subscription_sla: "Self-Support" + +## Check if access_ip responds to ping. Set false if your firewall blocks ICMP. +# ping_access_ip: true + +# sysctl_file_path to add sysctl conf to +# sysctl_file_path: "/etc/sysctl.d/99-sysctl.conf" + +## Variables for webhook token auth https://kubernetes.io/docs/reference/access-authn-authz/authentication/#webhook-token-authentication +kube_webhook_token_auth: false +kube_webhook_token_auth_url_skip_tls_verify: false +# kube_webhook_token_auth_url: https://... +## base64-encoded string of the webhook's CA certificate +# kube_webhook_token_auth_ca_data: "LS0t..." + +## NTP Settings +# Start the ntpd or chrony service and enable it at system boot. +ntp_enabled: false +ntp_manage_config: false +ntp_servers: + - "0.pool.ntp.org iburst" + - "1.pool.ntp.org iburst" + - "2.pool.ntp.org iburst" + - "3.pool.ntp.org iburst" + +## Used to control no_log attribute +unsafe_show_logs: false + +## If enabled it will allow kubespray to attempt setup even if the distribution is not supported. For unsupported distributions this can lead to unexpected failures in some cases. +allow_unsupported_distribution_setup: false diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/aws.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/aws.yml new file mode 100644 index 0000000..dab674e --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/aws.yml @@ -0,0 +1,9 @@ +## To use AWS EBS CSI Driver to provision volumes, uncomment the first value +## and configure the parameters below +# aws_ebs_csi_enabled: true +# aws_ebs_csi_enable_volume_scheduling: true +# aws_ebs_csi_enable_volume_snapshot: false +# aws_ebs_csi_enable_volume_resizing: false +# aws_ebs_csi_controller_replicas: 1 +# aws_ebs_csi_plugin_image_tag: latest +# aws_ebs_csi_extra_volume_tags: "Owner=owner,Team=team,Environment=environment' diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/azure.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/azure.yml new file mode 100644 index 0000000..176b0f1 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/azure.yml @@ -0,0 +1,40 @@ +## When azure is used, you need to also set the following variables. +## see docs/azure.md for details on how to get these values + +# azure_cloud: +# azure_tenant_id: +# azure_subscription_id: +# azure_aad_client_id: +# azure_aad_client_secret: +# azure_resource_group: +# azure_location: +# azure_subnet_name: +# azure_security_group_name: +# azure_security_group_resource_group: +# azure_vnet_name: +# azure_vnet_resource_group: +# azure_route_table_name: +# azure_route_table_resource_group: +# supported values are 'standard' or 'vmss' +# azure_vmtype: standard + +## Azure Disk CSI credentials and parameters +## see docs/azure-csi.md for details on how to get these values + +# azure_csi_tenant_id: +# azure_csi_subscription_id: +# azure_csi_aad_client_id: +# azure_csi_aad_client_secret: +# azure_csi_location: +# azure_csi_resource_group: +# azure_csi_vnet_name: +# azure_csi_vnet_resource_group: +# azure_csi_subnet_name: +# azure_csi_security_group_name: +# azure_csi_use_instance_metadata: +# azure_csi_tags: "Owner=owner,Team=team,Environment=environment' + +## To enable Azure Disk CSI, uncomment below +# azure_csi_enabled: true +# azure_csi_controller_replicas: 1 +# azure_csi_plugin_image_tag: latest diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/containerd.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/containerd.yml new file mode 100644 index 0000000..1888b24 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/containerd.yml @@ -0,0 +1,46 @@ +--- +# Please see roles/container-engine/containerd/defaults/main.yml for more configuration options + +# containerd_storage_dir: "/var/lib/containerd" +# containerd_state_dir: "/run/containerd" +# containerd_oom_score: 0 + +# containerd_default_runtime: "runc" +# containerd_snapshotter: "native" + +# containerd_runc_runtime: +# name: runc +# type: "io.containerd.runc.v2" +# engine: "" +# root: "" + +# containerd_additional_runtimes: +# Example for Kata Containers as additional runtime: +# - name: kata +# type: "io.containerd.kata.v2" +# engine: "" +# root: "" + +# containerd_grpc_max_recv_message_size: 16777216 +# containerd_grpc_max_send_message_size: 16777216 + +# containerd_debug_level: "info" + +# containerd_metrics_address: "" + +# containerd_metrics_grpc_histogram: false + +# Registries defined within containerd. +# containerd_registries_mirrors: +# - prefix: docker.io +# mirrors: +# - host: https://registry-1.docker.io +# capabilities: ["pull", "resolve"] +# skip_verify: false + +# containerd_max_container_log_line_size: -1 + +# containerd_registry_auth: +# - registry: 10.0.0.2:5000 +# username: user +# password: pass diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/coreos.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/coreos.yml new file mode 100644 index 0000000..22c2166 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/coreos.yml @@ -0,0 +1,2 @@ +## Does coreos need auto upgrade, default is true +# coreos_auto_upgrade: true diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/cri-o.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/cri-o.yml new file mode 100644 index 0000000..3e6e4ee --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/cri-o.yml @@ -0,0 +1,6 @@ +# crio_insecure_registries: +# - 10.0.0.2:5000 +# crio_registry_auth: +# - registry: 10.0.0.2:5000 +# username: user +# password: pass diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/docker.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/docker.yml new file mode 100644 index 0000000..4e968c3 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/docker.yml @@ -0,0 +1,59 @@ +--- +## Uncomment this if you want to force overlay/overlay2 as docker storage driver +## Please note that overlay2 is only supported on newer kernels +# docker_storage_options: -s overlay2 + +## Enable docker_container_storage_setup, it will configure devicemapper driver on Centos7 or RedHat7. +docker_container_storage_setup: false + +## It must be define a disk path for docker_container_storage_setup_devs. +## Otherwise docker-storage-setup will be executed incorrectly. +# docker_container_storage_setup_devs: /dev/vdb + +## Uncomment this if you want to change the Docker Cgroup driver (native.cgroupdriver) +## Valid options are systemd or cgroupfs, default is systemd +# docker_cgroup_driver: systemd + +## Only set this if you have more than 3 nameservers: +## If true Kubespray will only use the first 3, otherwise it will fail +docker_dns_servers_strict: false + +# Path used to store Docker data +docker_daemon_graph: "/var/lib/docker" + +## Used to set docker daemon iptables options to true +docker_iptables_enabled: "false" + +# Docker log options +# Rotate container stderr/stdout logs at 50m and keep last 5 +docker_log_opts: "--log-opt max-size=50m --log-opt max-file=5" + +# define docker bin_dir +docker_bin_dir: "/usr/bin" + +# keep docker packages after installation; speeds up repeated ansible provisioning runs when '1' +# kubespray deletes the docker package on each run, so caching the package makes sense +docker_rpm_keepcache: 1 + +## An obvious use case is allowing insecure-registry access to self hosted registries. +## Can be ipaddress and domain_name. +## example define 172.19.16.11 or mirror.registry.io +# docker_insecure_registries: +# - mirror.registry.io +# - 172.19.16.11 + +## Add other registry,example China registry mirror. +# docker_registry_mirrors: +# - https://registry.docker-cn.com +# - https://mirror.aliyuncs.com + +## If non-empty will override default system MountFlags value. +## This option takes a mount propagation flag: shared, slave +## or private, which control whether mounts in the file system +## namespace set up for docker will receive or propagate mounts +## and unmounts. Leave empty for system default +# docker_mount_flags: + +## A string of extra options to pass to the docker daemon. +## This string should be exactly as you wish it to appear. +# docker_options: "" diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/etcd.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/etcd.yml new file mode 100644 index 0000000..39600c3 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/etcd.yml @@ -0,0 +1,16 @@ +--- +## Directory where etcd data stored +etcd_data_dir: /var/lib/etcd + +## Container runtime +## docker for docker, crio for cri-o and containerd for containerd. +## Additionally you can set this to kubeadm if you want to install etcd using kubeadm +## Kubeadm etcd deployment is experimental and only available for new deployments +## If this is not set, container manager will be inherited from the Kubespray defaults +## and not from k8s_cluster/k8s-cluster.yml, which might not be what you want. +## Also this makes possible to use different container manager for etcd nodes. +# container_manager: containerd + +## Settings for etcd deployment type +# Set this to docker if you are using container_manager: docker +etcd_deployment_type: host diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/gcp.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/gcp.yml new file mode 100644 index 0000000..49eb5c0 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/gcp.yml @@ -0,0 +1,10 @@ +## GCP compute Persistent Disk CSI Driver credentials and parameters +## See docs/gcp-pd-csi.md for information about the implementation + +## Specify the path to the file containing the service account credentials +# gcp_pd_csi_sa_cred_file: "/my/safe/credentials/directory/cloud-sa.json" + +## To enable GCP Persistent Disk CSI driver, uncomment below +# gcp_pd_csi_enabled: true +# gcp_pd_csi_controller_replicas: 1 +# gcp_pd_csi_driver_image_tag: "v0.7.0-gke.0" diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/hcloud.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/hcloud.yml new file mode 100644 index 0000000..d4ed65c --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/hcloud.yml @@ -0,0 +1,22 @@ +## Values for the external Hcloud Cloud Controller +# external_hcloud_cloud: +# hcloud_api_token: "" +# token_secret_name: hcloud +# with_networks: false # Use the hcloud controller-manager with networks support https://github.com/hetznercloud/hcloud-cloud-controller-manager#networks-support +# network_name: # network name/ID: If you manage the network yourself it might still be required to let the CCM know about private networks +# service_account_name: cloud-controller-manager +# +# controller_image_tag: "latest" +# ## A dictionary of extra arguments to add to the openstack cloud controller manager daemonset +# ## Format: +# ## external_hcloud_cloud.controller_extra_args: +# ## arg1: "value1" +# ## arg2: "value2" +# controller_extra_args: {} +# +# load_balancers_location: # mutually exclusive with load_balancers_network_zone +# load_balancers_network_zone: +# load_balancers_disable_private_ingress: # set to true if using IPVS based plugins https://github.com/hetznercloud/hcloud-cloud-controller-manager/blob/main/docs/load_balancers.md#sample-service-with-networks +# load_balancers_use_private_ip: # set to true if using private networks +# load_balancers_enabled: +# network_routes_enabled: diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/huaweicloud.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/huaweicloud.yml new file mode 100644 index 0000000..20c7202 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/huaweicloud.yml @@ -0,0 +1,17 @@ +## Values for the external Huawei Cloud Controller +# external_huaweicloud_lbaas_subnet_id: "Neutron subnet ID to create LBaaS VIP" +# external_huaweicloud_lbaas_network_id: "Neutron network ID to create LBaaS VIP" + +## Credentials to authenticate against Keystone API +## All of them are required Per default these values will be +## read from the environment. +# external_huaweicloud_auth_url: "{{ lookup('env','OS_AUTH_URL') }}" +# external_huaweicloud_access_key: "{{ lookup('env','OS_ACCESS_KEY') }}" +# external_huaweicloud_secret_key: "{{ lookup('env','OS_SECRET_KEY') }}" +# external_huaweicloud_region: "{{ lookup('env','OS_REGION_NAME') }}" +# external_huaweicloud_project_id: "{{ lookup('env','OS_TENANT_ID')| default(lookup('env','OS_PROJECT_ID'),true) }}" +# external_huaweicloud_cloud: "{{ lookup('env','OS_CLOUD') }}" + +## The repo and tag of the external Huawei Cloud Controller image +# external_huawei_cloud_controller_image_repo: "swr.ap-southeast-1.myhuaweicloud.com" +# external_huawei_cloud_controller_image_tag: "v0.26.3" diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/oci.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/oci.yml new file mode 100644 index 0000000..541d0e6 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/oci.yml @@ -0,0 +1,28 @@ +## When Oracle Cloud Infrastructure is used, set these variables +# oci_private_key: +# oci_region_id: +# oci_tenancy_id: +# oci_user_id: +# oci_user_fingerprint: +# oci_compartment_id: +# oci_vnc_id: +# oci_subnet1_id: +# oci_subnet2_id: +## Override these default/optional behaviors if you wish +# oci_security_list_management: All +## If you would like the controller to manage specific lists per subnet. This is a mapping of subnet ocids to security list ocids. Below are examples. +# oci_security_lists: +# ocid1.subnet.oc1.phx.aaaaaaaasa53hlkzk6nzksqfccegk2qnkxmphkblst3riclzs4rhwg7rg57q: ocid1.securitylist.oc1.iad.aaaaaaaaqti5jsfvyw6ejahh7r4okb2xbtuiuguswhs746mtahn72r7adt7q +# ocid1.subnet.oc1.phx.aaaaaaaahuxrgvs65iwdz7ekwgg3l5gyah7ww5klkwjcso74u3e4i64hvtvq: ocid1.securitylist.oc1.iad.aaaaaaaaqti5jsfvyw6ejahh7r4okb2xbtuiuguswhs746mtahn72r7adt7q +## If oci_use_instance_principals is true, you do not need to set the region, tenancy, user, key, passphrase, or fingerprint +# oci_use_instance_principals: false +# oci_cloud_controller_version: 0.6.0 +## If you would like to control OCI query rate limits for the controller +# oci_rate_limit: +# rate_limit_qps_read: +# rate_limit_qps_write: +# rate_limit_bucket_read: +# rate_limit_bucket_write: +## Other optional variables +# oci_cloud_controller_pull_source: (default iad.ocir.io/oracle/cloud-provider-oci) +# oci_cloud_controller_pull_secret: (name of pull secret to use if you define your own mirror above) diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/offline.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/offline.yml new file mode 100644 index 0000000..7fba57e --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/offline.yml @@ -0,0 +1,106 @@ +--- +## Global Offline settings +### Private Container Image Registry +# registry_host: "myprivateregisry.com" +# files_repo: "http://myprivatehttpd" +### If using CentOS, RedHat, AlmaLinux or Fedora +# yum_repo: "http://myinternalyumrepo" +### If using Debian +# debian_repo: "http://myinternaldebianrepo" +### If using Ubuntu +# ubuntu_repo: "http://myinternalubunturepo" + +## Container Registry overrides +# kube_image_repo: "{{ registry_host }}" +# gcr_image_repo: "{{ registry_host }}" +# github_image_repo: "{{ registry_host }}" +# docker_image_repo: "{{ registry_host }}" +# quay_image_repo: "{{ registry_host }}" + +## Kubernetes components +# kubeadm_download_url: "{{ files_repo }}/dl.k8s.io/release/{{ kubeadm_version }}/bin/linux/{{ image_arch }}/kubeadm" +# kubectl_download_url: "{{ files_repo }}/dl.k8s.io/release/{{ kube_version }}/bin/linux/{{ image_arch }}/kubectl" +# kubelet_download_url: "{{ files_repo }}/dl.k8s.io/release/{{ kube_version }}/bin/linux/{{ image_arch }}/kubelet" + +## CNI Plugins +# cni_download_url: "{{ files_repo }}/github.com/containernetworking/plugins/releases/download/{{ cni_version }}/cni-plugins-linux-{{ image_arch }}-{{ cni_version }}.tgz" + +## cri-tools +# crictl_download_url: "{{ files_repo }}/github.com/kubernetes-sigs/cri-tools/releases/download/{{ crictl_version }}/crictl-{{ crictl_version }}-{{ ansible_system | lower }}-{{ image_arch }}.tar.gz" + +## [Optional] etcd: only if you use etcd_deployment=host +# etcd_download_url: "{{ files_repo }}/github.com/etcd-io/etcd/releases/download/{{ etcd_version }}/etcd-{{ etcd_version }}-linux-{{ image_arch }}.tar.gz" + +# [Optional] Calico: If using Calico network plugin +# calicoctl_download_url: "{{ files_repo }}/github.com/projectcalico/calico/releases/download/{{ calico_ctl_version }}/calicoctl-linux-{{ image_arch }}" +# [Optional] Calico with kdd: If using Calico network plugin with kdd datastore +# calico_crds_download_url: "{{ files_repo }}/github.com/projectcalico/calico/archive/{{ calico_version }}.tar.gz" + +# [Optional] Cilium: If using Cilium network plugin +# ciliumcli_download_url: "{{ files_repo }}/github.com/cilium/cilium-cli/releases/download/{{ cilium_cli_version }}/cilium-linux-{{ image_arch }}.tar.gz" + +# [Optional] helm: only if you set helm_enabled: true +# helm_download_url: "{{ files_repo }}/get.helm.sh/helm-{{ helm_version }}-linux-{{ image_arch }}.tar.gz" + +# [Optional] crun: only if you set crun_enabled: true +# crun_download_url: "{{ files_repo }}/github.com/containers/crun/releases/download/{{ crun_version }}/crun-{{ crun_version }}-linux-{{ image_arch }}" + +# [Optional] kata: only if you set kata_containers_enabled: true +# kata_containers_download_url: "{{ files_repo }}/github.com/kata-containers/kata-containers/releases/download/{{ kata_containers_version }}/kata-static-{{ kata_containers_version }}-{{ ansible_architecture }}.tar.xz" + +# [Optional] cri-dockerd: only if you set container_manager: docker +# cri_dockerd_download_url: "{{ files_repo }}/github.com/Mirantis/cri-dockerd/releases/download/v{{ cri_dockerd_version }}/cri-dockerd-{{ cri_dockerd_version }}.{{ image_arch }}.tgz" + +# [Optional] runc: if you set container_manager to containerd or crio +# runc_download_url: "{{ files_repo }}/github.com/opencontainers/runc/releases/download/{{ runc_version }}/runc.{{ image_arch }}" + +# [Optional] cri-o: only if you set container_manager: crio +# crio_download_base: "download.opensuse.org/repositories/devel:kubic:libcontainers:stable" +# crio_download_crio: "http://{{ crio_download_base }}:/cri-o:/" +# crio_download_url: "{{ files_repo }}/storage.googleapis.com/cri-o/artifacts/cri-o.{{ image_arch }}.{{ crio_version }}.tar.gz" +# skopeo_download_url: "{{ files_repo }}/github.com/lework/skopeo-binary/releases/download/{{ skopeo_version }}/skopeo-linux-{{ image_arch }}" + +# [Optional] containerd: only if you set container_runtime: containerd +# containerd_download_url: "{{ files_repo }}/github.com/containerd/containerd/releases/download/v{{ containerd_version }}/containerd-{{ containerd_version }}-linux-{{ image_arch }}.tar.gz" +# nerdctl_download_url: "{{ files_repo }}/github.com/containerd/nerdctl/releases/download/v{{ nerdctl_version }}/nerdctl-{{ nerdctl_version }}-{{ ansible_system | lower }}-{{ image_arch }}.tar.gz" + +# [Optional] runsc,containerd-shim-runsc: only if you set gvisor_enabled: true +# gvisor_runsc_download_url: "{{ files_repo }}/storage.googleapis.com/gvisor/releases/release/{{ gvisor_version }}/{{ ansible_architecture }}/runsc" +# gvisor_containerd_shim_runsc_download_url: "{{ files_repo }}/storage.googleapis.com/gvisor/releases/release/{{ gvisor_version }}/{{ ansible_architecture }}/containerd-shim-runsc-v1" + +# [Optional] Krew: only if you set krew_enabled: true +# krew_download_url: "{{ files_repo }}/github.com/kubernetes-sigs/krew/releases/download/{{ krew_version }}/krew-{{ host_os }}_{{ image_arch }}.tar.gz" + +## CentOS/Redhat/AlmaLinux +### For EL7, base and extras repo must be available, for EL8, baseos and appstream +### By default we enable those repo automatically +# rhel_enable_repos: false +### Docker / Containerd +# docker_rh_repo_base_url: "{{ yum_repo }}/docker-ce/$releasever/$basearch" +# docker_rh_repo_gpgkey: "{{ yum_repo }}/docker-ce/gpg" + +## Fedora +### Docker +# docker_fedora_repo_base_url: "{{ yum_repo }}/docker-ce/{{ ansible_distribution_major_version }}/{{ ansible_architecture }}" +# docker_fedora_repo_gpgkey: "{{ yum_repo }}/docker-ce/gpg" +### Containerd +# containerd_fedora_repo_base_url: "{{ yum_repo }}/containerd" +# containerd_fedora_repo_gpgkey: "{{ yum_repo }}/docker-ce/gpg" + +## Debian +### Docker +# docker_debian_repo_base_url: "{{ debian_repo }}/docker-ce" +# docker_debian_repo_gpgkey: "{{ debian_repo }}/docker-ce/gpg" +### Containerd +# containerd_debian_repo_base_url: "{{ debian_repo }}/containerd" +# containerd_debian_repo_gpgkey: "{{ debian_repo }}/containerd/gpg" +# containerd_debian_repo_repokey: 'YOURREPOKEY' + +## Ubuntu +### Docker +# docker_ubuntu_repo_base_url: "{{ ubuntu_repo }}/docker-ce" +# docker_ubuntu_repo_gpgkey: "{{ ubuntu_repo }}/docker-ce/gpg" +### Containerd +# containerd_ubuntu_repo_base_url: "{{ ubuntu_repo }}/containerd" +# containerd_ubuntu_repo_gpgkey: "{{ ubuntu_repo }}/containerd/gpg" +# containerd_ubuntu_repo_repokey: 'YOURREPOKEY' diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/openstack.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/openstack.yml new file mode 100644 index 0000000..0fec79a --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/openstack.yml @@ -0,0 +1,50 @@ +## When OpenStack is used, Cinder version can be explicitly specified if autodetection fails (Fixed in 1.9: https://github.com/kubernetes/kubernetes/issues/50461) +# openstack_blockstorage_version: "v1/v2/auto (default)" +# openstack_blockstorage_ignore_volume_az: yes +## When OpenStack is used, if LBaaSv2 is available you can enable it with the following 2 variables. +# openstack_lbaas_enabled: True +# openstack_lbaas_subnet_id: "Neutron subnet ID (not network ID) to create LBaaS VIP" +## To enable automatic floating ip provisioning, specify a subnet. +# openstack_lbaas_floating_network_id: "Neutron network ID (not subnet ID) to get floating IP from, disabled by default" +## Override default LBaaS behavior +# openstack_lbaas_use_octavia: False +# openstack_lbaas_method: "ROUND_ROBIN" +# openstack_lbaas_provider: "haproxy" +# openstack_lbaas_create_monitor: "yes" +# openstack_lbaas_monitor_delay: "1m" +# openstack_lbaas_monitor_timeout: "30s" +# openstack_lbaas_monitor_max_retries: "3" + +## Values for the external OpenStack Cloud Controller +# external_openstack_lbaas_enabled: true +# external_openstack_lbaas_floating_network_id: "Neutron network ID to get floating IP from" +# external_openstack_lbaas_floating_subnet_id: "Neutron subnet ID to get floating IP from" +# external_openstack_lbaas_method: ROUND_ROBIN +# external_openstack_lbaas_provider: amphora +# external_openstack_lbaas_subnet_id: "Neutron subnet ID to create LBaaS VIP" +# external_openstack_lbaas_network_id: "Neutron network ID to create LBaaS VIP" +# external_openstack_lbaas_manage_security_groups: false +# external_openstack_lbaas_create_monitor: false +# external_openstack_lbaas_monitor_delay: 5 +# external_openstack_lbaas_monitor_max_retries: 1 +# external_openstack_lbaas_monitor_timeout: 3 +# external_openstack_lbaas_internal_lb: false +# external_openstack_network_ipv6_disabled: false +# external_openstack_network_internal_networks: [] +# external_openstack_network_public_networks: [] +# external_openstack_metadata_search_order: "configDrive,metadataService" + +## Application credentials to authenticate against Keystone API +## Those settings will take precedence over username and password that might be set your environment +## All of them are required +# external_openstack_application_credential_name: +# external_openstack_application_credential_id: +# external_openstack_application_credential_secret: + +## The tag of the external OpenStack Cloud Controller image +# external_openstack_cloud_controller_image_tag: "latest" + +## To use Cinder CSI plugin to provision volumes set this value to true +## Make sure to source in the openstack credentials +# cinder_csi_enabled: true +# cinder_csi_controller_replicas: 1 diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/upcloud.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/upcloud.yml new file mode 100644 index 0000000..f05435d --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/upcloud.yml @@ -0,0 +1,24 @@ +## Repo for UpClouds csi-driver: https://github.com/UpCloudLtd/upcloud-csi +## To use UpClouds CSI plugin to provision volumes set this value to true +## Remember to set UPCLOUD_USERNAME and UPCLOUD_PASSWORD +# upcloud_csi_enabled: true +# upcloud_csi_controller_replicas: 1 +## Override used image tags +# upcloud_csi_provisioner_image_tag: "v3.1.0" +# upcloud_csi_attacher_image_tag: "v3.4.0" +# upcloud_csi_resizer_image_tag: "v1.4.0" +# upcloud_csi_plugin_image_tag: "v0.3.3" +# upcloud_csi_node_image_tag: "v2.5.0" +# upcloud_tolerations: [] +## Storage class options +# storage_classes: +# - name: standard +# is_default: true +# expand_persistent_volumes: true +# parameters: +# tier: maxiops +# - name: hdd +# is_default: false +# expand_persistent_volumes: true +# parameters: +# tier: hdd diff --git a/kubespray/inventory/moaroom-cluster/group_vars/all/vsphere.yml b/kubespray/inventory/moaroom-cluster/group_vars/all/vsphere.yml new file mode 100644 index 0000000..af3cfbe --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/all/vsphere.yml @@ -0,0 +1,32 @@ +## Values for the external vSphere Cloud Provider +# external_vsphere_vcenter_ip: "myvcenter.domain.com" +# external_vsphere_vcenter_port: "443" +# external_vsphere_insecure: "true" +# external_vsphere_user: "administrator@vsphere.local" # Can also be set via the `VSPHERE_USER` environment variable +# external_vsphere_password: "K8s_admin" # Can also be set via the `VSPHERE_PASSWORD` environment variable +# external_vsphere_datacenter: "DATACENTER_name" +# external_vsphere_kubernetes_cluster_id: "kubernetes-cluster-id" + +## Vsphere version where located VMs +# external_vsphere_version: "6.7u3" + +## Tags for the external vSphere Cloud Provider images +## gcr.io/cloud-provider-vsphere/cpi/release/manager +# external_vsphere_cloud_controller_image_tag: "latest" +## gcr.io/cloud-provider-vsphere/csi/release/syncer +# vsphere_syncer_image_tag: "v2.5.1" +## registry.k8s.io/sig-storage/csi-attacher +# vsphere_csi_attacher_image_tag: "v3.4.0" +## gcr.io/cloud-provider-vsphere/csi/release/driver +# vsphere_csi_controller: "v2.5.1" +## registry.k8s.io/sig-storage/livenessprobe +# vsphere_csi_liveness_probe_image_tag: "v2.6.0" +## registry.k8s.io/sig-storage/csi-provisioner +# vsphere_csi_provisioner_image_tag: "v3.1.0" +## registry.k8s.io/sig-storage/csi-resizer +## makes sense only for vSphere version >=7.0 +# vsphere_csi_resizer_tag: "v1.3.0" + +## To use vSphere CSI plugin to provision volumes set this value to true +# vsphere_csi_enabled: true +# vsphere_csi_controller_replicas: 1 diff --git a/kubespray/inventory/moaroom-cluster/group_vars/etcd.yml b/kubespray/inventory/moaroom-cluster/group_vars/etcd.yml new file mode 100644 index 0000000..f07c720 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/etcd.yml @@ -0,0 +1,26 @@ +--- +## Etcd auto compaction retention for mvcc key value store in hour +# etcd_compaction_retention: 0 + +## Set level of detail for etcd exported metrics, specify 'extensive' to include histogram metrics. +# etcd_metrics: basic + +## Etcd is restricted by default to 512M on systems under 4GB RAM, 512MB is not enough for much more than testing. +## Set this if your etcd nodes have less than 4GB but you want more RAM for etcd. Set to 0 for unrestricted RAM. +## This value is only relevant when deploying etcd with `etcd_deployment_type: docker` +# etcd_memory_limit: "512M" + +## Etcd has a default of 2G for its space quota. If you put a value in etcd_memory_limit which is less than +## etcd_quota_backend_bytes, you may encounter out of memory terminations of the etcd cluster. Please check +## etcd documentation for more information. +# 8G is a suggested maximum size for normal environments and etcd warns at startup if the configured value exceeds it. +# etcd_quota_backend_bytes: "2147483648" + +# Maximum client request size in bytes the server will accept. +# etcd is designed to handle small key value pairs typical for metadata. +# Larger requests will work, but may increase the latency of other requests +# etcd_max_request_bytes: "1572864" + +### ETCD: disable peer client cert authentication. +# This affects ETCD_PEER_CLIENT_CERT_AUTH variable +# etcd_peer_client_auth: true diff --git a/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/addons.yml b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/addons.yml new file mode 100644 index 0000000..38bc8ca --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/addons.yml @@ -0,0 +1,261 @@ +--- +# Kubernetes dashboard +# RBAC required. see docs/getting-started.md for access details. +# dashboard_enabled: false + +# Helm deployment +helm_enabled: false + +# Registry deployment +registry_enabled: false +# registry_namespace: kube-system +# registry_storage_class: "" +# registry_disk_size: "10Gi" + +# Metrics Server deployment +metrics_server_enabled: false +# metrics_server_container_port: 10250 +# metrics_server_kubelet_insecure_tls: true +# metrics_server_metric_resolution: 15s +# metrics_server_kubelet_preferred_address_types: "InternalIP,ExternalIP,Hostname" +# metrics_server_host_network: false +# metrics_server_replicas: 1 + +# Rancher Local Path Provisioner +local_path_provisioner_enabled: false +# local_path_provisioner_namespace: "local-path-storage" +# local_path_provisioner_storage_class: "local-path" +# local_path_provisioner_reclaim_policy: Delete +# local_path_provisioner_claim_root: /opt/local-path-provisioner/ +# local_path_provisioner_debug: false +# local_path_provisioner_image_repo: "rancher/local-path-provisioner" +# local_path_provisioner_image_tag: "v0.0.24" +# local_path_provisioner_helper_image_repo: "busybox" +# local_path_provisioner_helper_image_tag: "latest" + +# Local volume provisioner deployment +local_volume_provisioner_enabled: false +# local_volume_provisioner_namespace: kube-system +# local_volume_provisioner_nodelabels: +# - kubernetes.io/hostname +# - topology.kubernetes.io/region +# - topology.kubernetes.io/zone +# local_volume_provisioner_storage_classes: +# local-storage: +# host_dir: /mnt/disks +# mount_dir: /mnt/disks +# volume_mode: Filesystem +# fs_type: ext4 +# fast-disks: +# host_dir: /mnt/fast-disks +# mount_dir: /mnt/fast-disks +# block_cleaner_command: +# - "/scripts/shred.sh" +# - "2" +# volume_mode: Filesystem +# fs_type: ext4 +# local_volume_provisioner_tolerations: +# - effect: NoSchedule +# operator: Exists + +# CSI Volume Snapshot Controller deployment, set this to true if your CSI is able to manage snapshots +# currently, setting cinder_csi_enabled=true would automatically enable the snapshot controller +# Longhorn is an extenal CSI that would also require setting this to true but it is not included in kubespray +# csi_snapshot_controller_enabled: false +# csi snapshot namespace +# snapshot_controller_namespace: kube-system + +# CephFS provisioner deployment +cephfs_provisioner_enabled: false +# cephfs_provisioner_namespace: "cephfs-provisioner" +# cephfs_provisioner_cluster: ceph +# cephfs_provisioner_monitors: "172.24.0.1:6789,172.24.0.2:6789,172.24.0.3:6789" +# cephfs_provisioner_admin_id: admin +# cephfs_provisioner_secret: secret +# cephfs_provisioner_storage_class: cephfs +# cephfs_provisioner_reclaim_policy: Delete +# cephfs_provisioner_claim_root: /volumes +# cephfs_provisioner_deterministic_names: true + +# RBD provisioner deployment +rbd_provisioner_enabled: false +# rbd_provisioner_namespace: rbd-provisioner +# rbd_provisioner_replicas: 2 +# rbd_provisioner_monitors: "172.24.0.1:6789,172.24.0.2:6789,172.24.0.3:6789" +# rbd_provisioner_pool: kube +# rbd_provisioner_admin_id: admin +# rbd_provisioner_secret_name: ceph-secret-admin +# rbd_provisioner_secret: ceph-key-admin +# rbd_provisioner_user_id: kube +# rbd_provisioner_user_secret_name: ceph-secret-user +# rbd_provisioner_user_secret: ceph-key-user +# rbd_provisioner_user_secret_namespace: rbd-provisioner +# rbd_provisioner_fs_type: ext4 +# rbd_provisioner_image_format: "2" +# rbd_provisioner_image_features: layering +# rbd_provisioner_storage_class: rbd +# rbd_provisioner_reclaim_policy: Delete + +# Nginx ingress controller deployment +ingress_nginx_enabled: false +# ingress_nginx_host_network: false +ingress_publish_status_address: "" +# ingress_nginx_nodeselector: +# kubernetes.io/os: "linux" +# ingress_nginx_tolerations: +# - key: "node-role.kubernetes.io/master" +# operator: "Equal" +# value: "" +# effect: "NoSchedule" +# - key: "node-role.kubernetes.io/control-plane" +# operator: "Equal" +# value: "" +# effect: "NoSchedule" +# ingress_nginx_namespace: "ingress-nginx" +# ingress_nginx_insecure_port: 80 +# ingress_nginx_secure_port: 443 +# ingress_nginx_configmap: +# map-hash-bucket-size: "128" +# ssl-protocols: "TLSv1.2 TLSv1.3" +# ingress_nginx_configmap_tcp_services: +# 9000: "default/example-go:8080" +# ingress_nginx_configmap_udp_services: +# 53: "kube-system/coredns:53" +# ingress_nginx_extra_args: +# - --default-ssl-certificate=default/foo-tls +# ingress_nginx_termination_grace_period_seconds: 300 +# ingress_nginx_class: nginx +# ingress_nginx_without_class: true +# ingress_nginx_default: false + +# ALB ingress controller deployment +ingress_alb_enabled: false +# alb_ingress_aws_region: "us-east-1" +# alb_ingress_restrict_scheme: "false" +# Enables logging on all outbound requests sent to the AWS API. +# If logging is desired, set to true. +# alb_ingress_aws_debug: "false" + +# Cert manager deployment +cert_manager_enabled: false +# cert_manager_namespace: "cert-manager" +# cert_manager_tolerations: +# - key: node-role.kubernetes.io/master +# effect: NoSchedule +# - key: node-role.kubernetes.io/control-plane +# effect: NoSchedule +# cert_manager_affinity: +# nodeAffinity: +# preferredDuringSchedulingIgnoredDuringExecution: +# - weight: 100 +# preference: +# matchExpressions: +# - key: node-role.kubernetes.io/control-plane +# operator: In +# values: +# - "" +# cert_manager_nodeselector: +# kubernetes.io/os: "linux" + +# cert_manager_trusted_internal_ca: | +# -----BEGIN CERTIFICATE----- +# [REPLACE with your CA certificate] +# -----END CERTIFICATE----- +# cert_manager_leader_election_namespace: kube-system + +# cert_manager_dns_policy: "ClusterFirst" +# cert_manager_dns_config: +# nameservers: +# - "1.1.1.1" +# - "8.8.8.8" + +# cert_manager_controller_extra_args: +# - "--dns01-recursive-nameservers-only=true" +# - "--dns01-recursive-nameservers=1.1.1.1:53,8.8.8.8:53" + +# MetalLB deployment +metallb_enabled: false +metallb_speaker_enabled: "{{ metallb_enabled }}" +# metallb_speaker_nodeselector: +# kubernetes.io/os: "linux" +# metallb_controller_nodeselector: +# kubernetes.io/os: "linux" +# metallb_speaker_tolerations: +# - key: "node-role.kubernetes.io/master" +# operator: "Equal" +# value: "" +# effect: "NoSchedule" +# - key: "node-role.kubernetes.io/control-plane" +# operator: "Equal" +# value: "" +# effect: "NoSchedule" +# metallb_controller_tolerations: +# - key: "node-role.kubernetes.io/master" +# operator: "Equal" +# value: "" +# effect: "NoSchedule" +# - key: "node-role.kubernetes.io/control-plane" +# operator: "Equal" +# value: "" +# effect: "NoSchedule" +# metallb_version: v0.13.9 +# metallb_protocol: "layer2" +# metallb_port: "7472" +# metallb_memberlist_port: "7946" +# metallb_config: +# address_pools: +# primary: +# ip_range: +# - 10.5.0.0/16 +# auto_assign: true +# pool1: +# ip_range: +# - 10.6.0.0/16 +# auto_assign: true +# pool2: +# ip_range: +# - 10.10.0.0/16 +# auto_assign: true +# layer2: +# - primary +# layer3: +# defaults: +# peer_port: 179 +# hold_time: 120s +# communities: +# vpn-only: "1234:1" +# NO_ADVERTISE: "65535:65282" +# metallb_peers: +# peer1: +# peer_address: 10.6.0.1 +# peer_asn: 64512 +# my_asn: 4200000000 +# communities: +# - vpn-only +# address_pool: +# - pool1 +# peer2: +# peer_address: 10.10.0.1 +# peer_asn: 64513 +# my_asn: 4200000000 +# communities: +# - NO_ADVERTISE +# address_pool: +# - pool2 + +argocd_enabled: false +# argocd_version: v2.8.0 +# argocd_namespace: argocd +# Default password: +# - https://argo-cd.readthedocs.io/en/stable/getting_started/#4-login-using-the-cli +# --- +# The initial password is autogenerated and stored in `argocd-initial-admin-secret` in the argocd namespace defined above. +# Using the argocd CLI the generated password can be automatically be fetched from the current kubectl context with the command: +# argocd admin initial-password -n argocd +# --- +# Use the following var to set admin password +# argocd_admin_password: "password" + +# The plugin manager for kubectl +krew_enabled: false +krew_root_dir: "/usr/local/krew" diff --git a/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-cluster.yml b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-cluster.yml new file mode 100644 index 0000000..69574c8 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-cluster.yml @@ -0,0 +1,382 @@ +--- +# Kubernetes configuration dirs and system namespace. +# Those are where all the additional config stuff goes +# the kubernetes normally puts in /srv/kubernetes. +# This puts them in a sane location and namespace. +# Editing those values will almost surely break something. +kube_config_dir: /etc/kubernetes +kube_script_dir: "{{ bin_dir }}/kubernetes-scripts" +kube_manifest_dir: "{{ kube_config_dir }}/manifests" + +# This is where all the cert scripts and certs will be located +kube_cert_dir: "{{ kube_config_dir }}/ssl" + +# This is where all of the bearer tokens will be stored +kube_token_dir: "{{ kube_config_dir }}/tokens" + +kube_api_anonymous_auth: true + +## Change this to use another Kubernetes version, e.g. a current beta release +kube_version: v1.28.2 + +# Where the binaries will be downloaded. +# Note: ensure that you've enough disk space (about 1G) +local_release_dir: "/tmp/releases" +# Random shifts for retrying failed ops like pushing/downloading +retry_stagger: 5 + +# This is the user that owns tha cluster installation. +kube_owner: kube + +# This is the group that the cert creation scripts chgrp the +# cert files to. Not really changeable... +kube_cert_group: kube-cert + +# Cluster Loglevel configuration +kube_log_level: 2 + +# Directory where credentials will be stored +credentials_dir: "{{ inventory_dir }}/credentials" + +## It is possible to activate / deactivate selected authentication methods (oidc, static token auth) +# kube_oidc_auth: false +# kube_token_auth: false + + +## Variables for OpenID Connect Configuration https://kubernetes.io/docs/admin/authentication/ +## To use OpenID you have to deploy additional an OpenID Provider (e.g Dex, Keycloak, ...) + +# kube_oidc_url: https:// ... +# kube_oidc_client_id: kubernetes +## Optional settings for OIDC +# kube_oidc_ca_file: "{{ kube_cert_dir }}/ca.pem" +# kube_oidc_username_claim: sub +# kube_oidc_username_prefix: 'oidc:' +# kube_oidc_groups_claim: groups +# kube_oidc_groups_prefix: 'oidc:' + +## Variables to control webhook authn/authz +# kube_webhook_token_auth: false +# kube_webhook_token_auth_url: https://... +# kube_webhook_token_auth_url_skip_tls_verify: false + +## For webhook authorization, authorization_modes must include Webhook +# kube_webhook_authorization: false +# kube_webhook_authorization_url: https://... +# kube_webhook_authorization_url_skip_tls_verify: false + +# Choose network plugin (cilium, calico, kube-ovn, weave or flannel. Use cni for generic cni plugin) +# Can also be set to 'cloud', which lets the cloud provider setup appropriate routing +kube_network_plugin: calico + +# Setting multi_networking to true will install Multus: https://github.com/k8snetworkplumbingwg/multus-cni +kube_network_plugin_multus: false + +# Kubernetes internal network for services, unused block of space. +kube_service_addresses: 10.233.0.0/18 + +# internal network. When used, it will assign IP +# addresses from this range to individual pods. +# This network must be unused in your network infrastructure! +kube_pods_subnet: 10.233.64.0/18 + +# internal network node size allocation (optional). This is the size allocated +# to each node for pod IP address allocation. Note that the number of pods per node is +# also limited by the kubelet_max_pods variable which defaults to 110. +# +# Example: +# Up to 64 nodes and up to 254 or kubelet_max_pods (the lowest of the two) pods per node: +# - kube_pods_subnet: 10.233.64.0/18 +# - kube_network_node_prefix: 24 +# - kubelet_max_pods: 110 +# +# Example: +# Up to 128 nodes and up to 126 or kubelet_max_pods (the lowest of the two) pods per node: +# - kube_pods_subnet: 10.233.64.0/18 +# - kube_network_node_prefix: 25 +# - kubelet_max_pods: 110 +kube_network_node_prefix: 24 + +# Configure Dual Stack networking (i.e. both IPv4 and IPv6) +enable_dual_stack_networks: false + +# Kubernetes internal network for IPv6 services, unused block of space. +# This is only used if enable_dual_stack_networks is set to true +# This provides 4096 IPv6 IPs +kube_service_addresses_ipv6: fd85:ee78:d8a6:8607::1000/116 + +# Internal network. When used, it will assign IPv6 addresses from this range to individual pods. +# This network must not already be in your network infrastructure! +# This is only used if enable_dual_stack_networks is set to true. +# This provides room for 256 nodes with 254 pods per node. +kube_pods_subnet_ipv6: fd85:ee78:d8a6:8607::1:0000/112 + +# IPv6 subnet size allocated to each for pods. +# This is only used if enable_dual_stack_networks is set to true +# This provides room for 254 pods per node. +kube_network_node_prefix_ipv6: 120 + +# The port the API Server will be listening on. +kube_apiserver_ip: "{{ kube_service_addresses | ipaddr('net') | ipaddr(1) | ipaddr('address') }}" +kube_apiserver_port: 6443 # (https) + +# Kube-proxy proxyMode configuration. +# Can be ipvs, iptables +kube_proxy_mode: ipvs + +# configure arp_ignore and arp_announce to avoid answering ARP queries from kube-ipvs0 interface +# must be set to true for MetalLB, kube-vip(ARP enabled) to work +kube_proxy_strict_arp: false + +# A string slice of values which specify the addresses to use for NodePorts. +# Values may be valid IP blocks (e.g. 1.2.3.0/24, 1.2.3.4/32). +# The default empty string slice ([]) means to use all local addresses. +# kube_proxy_nodeport_addresses_cidr is retained for legacy config +kube_proxy_nodeport_addresses: >- + {%- if kube_proxy_nodeport_addresses_cidr is defined -%} + [{{ kube_proxy_nodeport_addresses_cidr }}] + {%- else -%} + [] + {%- endif -%} + +# If non-empty, will use this string as identification instead of the actual hostname +# kube_override_hostname: >- +# {%- if cloud_provider is defined and cloud_provider in ['aws'] -%} +# {%- else -%} +# {{ inventory_hostname }} +# {%- endif -%} + +## Encrypting Secret Data at Rest +kube_encrypt_secret_data: false + +# Graceful Node Shutdown (Kubernetes >= 1.21.0), see https://kubernetes.io/blog/2021/04/21/graceful-node-shutdown-beta/ +# kubelet_shutdown_grace_period had to be greater than kubelet_shutdown_grace_period_critical_pods to allow +# non-critical podsa to also terminate gracefully +# kubelet_shutdown_grace_period: 60s +# kubelet_shutdown_grace_period_critical_pods: 20s + +# DNS configuration. +# Kubernetes cluster name, also will be used as DNS domain +cluster_name: cluster.local +# Subdomains of DNS domain to be resolved via /etc/resolv.conf for hostnet pods +ndots: 2 +# dns_timeout: 2 +# dns_attempts: 2 +# Custom search domains to be added in addition to the default cluster search domains +# searchdomains: +# - svc.{{ cluster_name }} +# - default.svc.{{ cluster_name }} +# Remove default cluster search domains (``default.svc.{{ dns_domain }}, svc.{{ dns_domain }}``). +# remove_default_searchdomains: false +# Can be coredns, coredns_dual, manual or none +dns_mode: coredns +# Set manual server if using a custom cluster DNS server +# manual_dns_server: 10.x.x.x +# Enable nodelocal dns cache +enable_nodelocaldns: true +enable_nodelocaldns_secondary: false +nodelocaldns_ip: 169.254.25.10 +nodelocaldns_health_port: 9254 +nodelocaldns_second_health_port: 9256 +nodelocaldns_bind_metrics_host_ip: false +nodelocaldns_secondary_skew_seconds: 5 +# nodelocaldns_external_zones: +# - zones: +# - example.com +# - example.io:1053 +# nameservers: +# - 1.1.1.1 +# - 2.2.2.2 +# cache: 5 +# - zones: +# - https://mycompany.local:4453 +# nameservers: +# - 192.168.0.53 +# cache: 0 +# - zones: +# - mydomain.tld +# nameservers: +# - 10.233.0.3 +# cache: 5 +# rewrite: +# - name website.tld website.namespace.svc.cluster.local +# Enable k8s_external plugin for CoreDNS +enable_coredns_k8s_external: false +coredns_k8s_external_zone: k8s_external.local +# Enable endpoint_pod_names option for kubernetes plugin +enable_coredns_k8s_endpoint_pod_names: false +# Set forward options for upstream DNS servers in coredns (and nodelocaldns) config +# dns_upstream_forward_extra_opts: +# policy: sequential +# Apply extra options to coredns kubernetes plugin +# coredns_kubernetes_extra_opts: +# - 'fallthrough example.local' +# Forward extra domains to the coredns kubernetes plugin +# coredns_kubernetes_extra_domains: '' + +# Can be docker_dns, host_resolvconf or none +resolvconf_mode: host_resolvconf +# Deploy netchecker app to verify DNS resolve as an HTTP service +deploy_netchecker: false +# Ip address of the kubernetes skydns service +skydns_server: "{{ kube_service_addresses | ipaddr('net') | ipaddr(3) | ipaddr('address') }}" +skydns_server_secondary: "{{ kube_service_addresses | ipaddr('net') | ipaddr(4) | ipaddr('address') }}" +dns_domain: "{{ cluster_name }}" + +## Container runtime +## docker for docker, crio for cri-o and containerd for containerd. +## Default: containerd +container_manager: containerd + +# Additional container runtimes +kata_containers_enabled: false + +kubeadm_certificate_key: "{{ lookup('password', credentials_dir + '/kubeadm_certificate_key.creds length=64 chars=hexdigits') | lower }}" + +# K8s image pull policy (imagePullPolicy) +k8s_image_pull_policy: IfNotPresent + +# audit log for kubernetes +kubernetes_audit: false + +# define kubelet config dir for dynamic kubelet +# kubelet_config_dir: +default_kubelet_config_dir: "{{ kube_config_dir }}/dynamic_kubelet_dir" + +# pod security policy (RBAC must be enabled either by having 'RBAC' in authorization_modes or kubeadm enabled) +podsecuritypolicy_enabled: false + +# Custom PodSecurityPolicySpec for restricted policy +# podsecuritypolicy_restricted_spec: {} + +# Custom PodSecurityPolicySpec for privileged policy +# podsecuritypolicy_privileged_spec: {} + +# Make a copy of kubeconfig on the host that runs Ansible in {{ inventory_dir }}/artifacts +# kubeconfig_localhost: false +# Use ansible_host as external api ip when copying over kubeconfig. +# kubeconfig_localhost_ansible_host: false +# Download kubectl onto the host that runs Ansible in {{ bin_dir }} +# kubectl_localhost: false + +# A comma separated list of levels of node allocatable enforcement to be enforced by kubelet. +# Acceptable options are 'pods', 'system-reserved', 'kube-reserved' and ''. Default is "". +# kubelet_enforce_node_allocatable: pods + +## Set runtime and kubelet cgroups when using systemd as cgroup driver (default) +# kubelet_runtime_cgroups: "/{{ kube_service_cgroups }}/{{ container_manager }}.service" +# kubelet_kubelet_cgroups: "/{{ kube_service_cgroups }}/kubelet.service" + +## Set runtime and kubelet cgroups when using cgroupfs as cgroup driver +# kubelet_runtime_cgroups_cgroupfs: "/system.slice/{{ container_manager }}.service" +# kubelet_kubelet_cgroups_cgroupfs: "/system.slice/kubelet.service" + +# Optionally reserve this space for kube daemons. +# kube_reserved: false +## Uncomment to override default values +## The following two items need to be set when kube_reserved is true +# kube_reserved_cgroups_for_service_slice: kube.slice +# kube_reserved_cgroups: "/{{ kube_reserved_cgroups_for_service_slice }}" +# kube_memory_reserved: 256Mi +# kube_cpu_reserved: 100m +# kube_ephemeral_storage_reserved: 2Gi +# kube_pid_reserved: "1000" +# Reservation for master hosts +# kube_master_memory_reserved: 512Mi +# kube_master_cpu_reserved: 200m +# kube_master_ephemeral_storage_reserved: 2Gi +# kube_master_pid_reserved: "1000" + +## Optionally reserve resources for OS system daemons. +# system_reserved: true +## Uncomment to override default values +## The following two items need to be set when system_reserved is true +# system_reserved_cgroups_for_service_slice: system.slice +# system_reserved_cgroups: "/{{ system_reserved_cgroups_for_service_slice }}" +# system_memory_reserved: 512Mi +# system_cpu_reserved: 500m +# system_ephemeral_storage_reserved: 2Gi +## Reservation for master hosts +# system_master_memory_reserved: 256Mi +# system_master_cpu_reserved: 250m +# system_master_ephemeral_storage_reserved: 2Gi + +## Eviction Thresholds to avoid system OOMs +# https://kubernetes.io/docs/tasks/administer-cluster/reserve-compute-resources/#eviction-thresholds +# eviction_hard: {} +# eviction_hard_control_plane: {} + +# An alternative flexvolume plugin directory +# kubelet_flexvolumes_plugins_dir: /usr/libexec/kubernetes/kubelet-plugins/volume/exec + +## Supplementary addresses that can be added in kubernetes ssl keys. +## That can be useful for example to setup a keepalived virtual IP +# supplementary_addresses_in_ssl_keys: [10.0.0.1, 10.0.0.2, 10.0.0.3] + +## Running on top of openstack vms with cinder enabled may lead to unschedulable pods due to NoVolumeZoneConflict restriction in kube-scheduler. +## See https://github.com/kubernetes-sigs/kubespray/issues/2141 +## Set this variable to true to get rid of this issue +volume_cross_zone_attachment: false +## Add Persistent Volumes Storage Class for corresponding cloud provider (supported: in-tree OpenStack, Cinder CSI, +## AWS EBS CSI, Azure Disk CSI, GCP Persistent Disk CSI) +persistent_volumes_enabled: false + +## Container Engine Acceleration +## Enable container acceleration feature, for example use gpu acceleration in containers +# nvidia_accelerator_enabled: true +## Nvidia GPU driver install. Install will by done by a (init) pod running as a daemonset. +## Important: if you use Ubuntu then you should set in all.yml 'docker_storage_options: -s overlay2' +## Array with nvida_gpu_nodes, leave empty or comment if you don't want to install drivers. +## Labels and taints won't be set to nodes if they are not in the array. +# nvidia_gpu_nodes: +# - kube-gpu-001 +# nvidia_driver_version: "384.111" +## flavor can be tesla or gtx +# nvidia_gpu_flavor: gtx +## NVIDIA driver installer images. Change them if you have trouble accessing gcr.io. +# nvidia_driver_install_centos_container: atzedevries/nvidia-centos-driver-installer:2 +# nvidia_driver_install_ubuntu_container: gcr.io/google-containers/ubuntu-nvidia-driver-installer@sha256:7df76a0f0a17294e86f691c81de6bbb7c04a1b4b3d4ea4e7e2cccdc42e1f6d63 +## NVIDIA GPU device plugin image. +# nvidia_gpu_device_plugin_container: "registry.k8s.io/nvidia-gpu-device-plugin@sha256:0842734032018be107fa2490c98156992911e3e1f2a21e059ff0105b07dd8e9e" + +## Support tls min version, Possible values: VersionTLS10, VersionTLS11, VersionTLS12, VersionTLS13. +# tls_min_version: "" + +## Support tls cipher suites. +# tls_cipher_suites: {} +# - TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA +# - TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256 +# - TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 +# - TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA +# - TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 +# - TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305 +# - TLS_ECDHE_ECDSA_WITH_RC4_128_SHA +# - TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256 +# - TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 +# - TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 +# - TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305 +# - TLS_ECDHE_RSA_WITH_RC4_128_SHA +# - TLS_RSA_WITH_3DES_EDE_CBC_SHA +# - TLS_RSA_WITH_AES_128_CBC_SHA +# - TLS_RSA_WITH_AES_128_CBC_SHA256 +# - TLS_RSA_WITH_AES_128_GCM_SHA256 +# - TLS_RSA_WITH_AES_256_CBC_SHA +# - TLS_RSA_WITH_AES_256_GCM_SHA384 +# - TLS_RSA_WITH_RC4_128_SHA + +## Amount of time to retain events. (default 1h0m0s) +event_ttl_duration: "1h0m0s" + +## Automatically renew K8S control plane certificates on first Monday of each month +auto_renew_certificates: false +# First Monday of each month +# auto_renew_certificates_systemd_calendar: "Mon *-*-1,2,3,4,5,6,7 03:{{ groups['kube_control_plane'].index(inventory_hostname) }}0:00" + +# kubeadm patches path +kubeadm_patches: + enabled: false + source_dir: "{{ inventory_dir }}/patches" + dest_dir: "{{ kube_config_dir }}/patches" diff --git a/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-calico.yml b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-calico.yml new file mode 100644 index 0000000..cc0499d --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-calico.yml @@ -0,0 +1,131 @@ +--- +# see roles/network_plugin/calico/defaults/main.yml + +# the default value of name +calico_cni_name: k8s-pod-network + +## With calico it is possible to distributed routes with border routers of the datacenter. +## Warning : enabling router peering will disable calico's default behavior ('node mesh'). +## The subnets of each nodes will be distributed by the datacenter router +# peer_with_router: false + +# Enables Internet connectivity from containers +# nat_outgoing: true + +# Enables Calico CNI "host-local" IPAM plugin +# calico_ipam_host_local: true + +# add default ippool name +# calico_pool_name: "default-pool" + +# add default ippool blockSize (defaults kube_network_node_prefix) +calico_pool_blocksize: 26 + +# add default ippool CIDR (must be inside kube_pods_subnet, defaults to kube_pods_subnet otherwise) +# calico_pool_cidr: 1.2.3.4/5 + +# add default ippool CIDR to CNI config +# calico_cni_pool: true + +# Add default IPV6 IPPool CIDR. Must be inside kube_pods_subnet_ipv6. Defaults to kube_pods_subnet_ipv6 if not set. +# calico_pool_cidr_ipv6: fd85:ee78:d8a6:8607::1:0000/112 + +# Add default IPV6 IPPool CIDR to CNI config +# calico_cni_pool_ipv6: true + +# Global as_num (/calico/bgp/v1/global/as_num) +# global_as_num: "64512" + +# If doing peering with node-assigned asn where the globas does not match your nodes, you want this +# to be true. All other cases, false. +# calico_no_global_as_num: false + +# You can set MTU value here. If left undefined or empty, it will +# not be specified in calico CNI config, so Calico will use built-in +# defaults. The value should be a number, not a string. +# calico_mtu: 1500 + +# Configure the MTU to use for workload interfaces and tunnels. +# - If Wireguard is enabled, subtract 60 from your network MTU (i.e 1500-60=1440) +# - Otherwise, if VXLAN or BPF mode is enabled, subtract 50 from your network MTU (i.e. 1500-50=1450) +# - Otherwise, if IPIP is enabled, subtract 20 from your network MTU (i.e. 1500-20=1480) +# - Otherwise, if not using any encapsulation, set to your network MTU (i.e. 1500) +# calico_veth_mtu: 1440 + +# Advertise Cluster IPs +# calico_advertise_cluster_ips: true + +# Advertise Service External IPs +# calico_advertise_service_external_ips: +# - x.x.x.x/24 +# - y.y.y.y/32 + +# Advertise Service LoadBalancer IPs +# calico_advertise_service_loadbalancer_ips: +# - x.x.x.x/24 +# - y.y.y.y/16 + +# Choose data store type for calico: "etcd" or "kdd" (kubernetes datastore) +# calico_datastore: "kdd" + +# Choose Calico iptables backend: "Legacy", "Auto" or "NFT" +# calico_iptables_backend: "Auto" + +# Use typha (only with kdd) +# typha_enabled: false + +# Generate TLS certs for secure typha<->calico-node communication +# typha_secure: false + +# Scaling typha: 1 replica per 100 nodes is adequate +# Number of typha replicas +# typha_replicas: 1 + +# Set max typha connections +# typha_max_connections_lower_limit: 300 + +# Set calico network backend: "bird", "vxlan" or "none" +# bird enable BGP routing, required for ipip and no encapsulation modes +# calico_network_backend: vxlan + +# IP in IP and VXLAN is mutualy exclusive modes. +# set IP in IP encapsulation mode: "Always", "CrossSubnet", "Never" +# calico_ipip_mode: 'Never' + +# set VXLAN encapsulation mode: "Always", "CrossSubnet", "Never" +# calico_vxlan_mode: 'Always' + +# set VXLAN port and VNI +# calico_vxlan_vni: 4096 +# calico_vxlan_port: 4789 + +# Enable eBPF mode +# calico_bpf_enabled: false + +# If you want to use non default IP_AUTODETECTION_METHOD, IP6_AUTODETECTION_METHOD for calico node set this option to one of: +# * can-reach=DESTINATION +# * interface=INTERFACE-REGEX +# see https://docs.projectcalico.org/reference/node/configuration +# calico_ip_auto_method: "interface=eth.*" +# calico_ip6_auto_method: "interface=eth.*" + +# Set FELIX_MTUIFACEPATTERN, Pattern used to discover the host’s interface for MTU auto-detection. +# see https://projectcalico.docs.tigera.io/reference/felix/configuration +# calico_felix_mtu_iface_pattern: "^((en|wl|ww|sl|ib)[opsx].*|(eth|wlan|wwan).*)" + +# Choose the iptables insert mode for Calico: "Insert" or "Append". +# calico_felix_chaininsertmode: Insert + +# If you want use the default route interface when you use multiple interface with dynamique route (iproute2) +# see https://docs.projectcalico.org/reference/node/configuration : FELIX_DEVICEROUTESOURCEADDRESS +# calico_use_default_route_src_ipaddr: false + +# Enable calico traffic encryption with wireguard +# calico_wireguard_enabled: false + +# Under certain situations liveness and readiness probes may need tunning +# calico_node_livenessprobe_timeout: 10 +# calico_node_readinessprobe_timeout: 10 + +# Calico apiserver (only with kdd) +# calico_apiserver_enabled: false diff --git a/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-cilium.yml b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-cilium.yml new file mode 100644 index 0000000..a170484 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-cilium.yml @@ -0,0 +1,264 @@ +--- +# cilium_version: "v1.12.1" + +# Log-level +# cilium_debug: false + +# cilium_mtu: "" +# cilium_enable_ipv4: true +# cilium_enable_ipv6: false + +# Cilium agent health port +# cilium_agent_health_port: "9879" + +# Identity allocation mode selects how identities are shared between cilium +# nodes by setting how they are stored. The options are "crd" or "kvstore". +# - "crd" stores identities in kubernetes as CRDs (custom resource definition). +# These can be queried with: +# `kubectl get ciliumid` +# - "kvstore" stores identities in an etcd kvstore. +# - In order to support External Workloads, "crd" is required +# - Ref: https://docs.cilium.io/en/stable/gettingstarted/external-workloads/#setting-up-support-for-external-workloads-beta +# - KVStore operations are only required when cilium-operator is running with any of the below options: +# - --synchronize-k8s-services +# - --synchronize-k8s-nodes +# - --identity-allocation-mode=kvstore +# - Ref: https://docs.cilium.io/en/stable/internals/cilium_operator/#kvstore-operations +# cilium_identity_allocation_mode: kvstore + +# Etcd SSL dirs +# cilium_cert_dir: /etc/cilium/certs +# kube_etcd_cacert_file: ca.pem +# kube_etcd_cert_file: cert.pem +# kube_etcd_key_file: cert-key.pem + +# Limits for apps +# cilium_memory_limit: 500M +# cilium_cpu_limit: 500m +# cilium_memory_requests: 64M +# cilium_cpu_requests: 100m + +# Overlay Network Mode +# cilium_tunnel_mode: vxlan +# Optional features +# cilium_enable_prometheus: false +# Enable if you want to make use of hostPort mappings +# cilium_enable_portmap: false +# Monitor aggregation level (none/low/medium/maximum) +# cilium_monitor_aggregation: medium +# The monitor aggregation flags determine which TCP flags which, upon the +# first observation, cause monitor notifications to be generated. +# +# Only effective when monitor aggregation is set to "medium" or higher. +# cilium_monitor_aggregation_flags: "all" +# Kube Proxy Replacement mode (strict/partial) +# cilium_kube_proxy_replacement: partial + +# If upgrading from Cilium < 1.5, you may want to override some of these options +# to prevent service disruptions. See also: +# http://docs.cilium.io/en/stable/install/upgrade/#changes-that-may-require-action +# cilium_preallocate_bpf_maps: false + +# `cilium_tofqdns_enable_poller` is deprecated in 1.8, removed in 1.9 +# cilium_tofqdns_enable_poller: false + +# `cilium_enable_legacy_services` is deprecated in 1.6, removed in 1.9 +# cilium_enable_legacy_services: false + +# Unique ID of the cluster. Must be unique across all conneted clusters and +# in the range of 1 and 255. Only relevant when building a mesh of clusters. +# This value is not defined by default +# cilium_cluster_id: + +# Deploy cilium even if kube_network_plugin is not cilium. +# This enables to deploy cilium alongside another CNI to replace kube-proxy. +# cilium_deploy_additionally: false + +# Auto direct nodes routes can be used to advertise pods routes in your cluster +# without any tunelling (with `cilium_tunnel_mode` sets to `disabled`). +# This works only if you have a L2 connectivity between all your nodes. +# You wil also have to specify the variable `cilium_native_routing_cidr` to +# make this work. Please refer to the cilium documentation for more +# information about this kind of setups. +# cilium_auto_direct_node_routes: false + +# Allows to explicitly specify the IPv4 CIDR for native routing. +# When specified, Cilium assumes networking for this CIDR is preconfigured and +# hands traffic destined for that range to the Linux network stack without +# applying any SNAT. +# Generally speaking, specifying a native routing CIDR implies that Cilium can +# depend on the underlying networking stack to route packets to their +# destination. To offer a concrete example, if Cilium is configured to use +# direct routing and the Kubernetes CIDR is included in the native routing CIDR, +# the user must configure the routes to reach pods, either manually or by +# setting the auto-direct-node-routes flag. +# cilium_native_routing_cidr: "" + +# Allows to explicitly specify the IPv6 CIDR for native routing. +# cilium_native_routing_cidr_ipv6: "" + +# Enable transparent network encryption. +# cilium_encryption_enabled: false + +# Encryption method. Can be either ipsec or wireguard. +# Only effective when `cilium_encryption_enabled` is set to true. +# cilium_encryption_type: "ipsec" + +# Enable encryption for pure node to node traffic. +# This option is only effective when `cilium_encryption_type` is set to `ipsec`. +# cilium_ipsec_node_encryption: false + +# If your kernel or distribution does not support WireGuard, Cilium agent can be configured to fall back on the user-space implementation. +# When this flag is enabled and Cilium detects that the kernel has no native support for WireGuard, +# it will fallback on the wireguard-go user-space implementation of WireGuard. +# This option is only effective when `cilium_encryption_type` is set to `wireguard`. +# cilium_wireguard_userspace_fallback: false + +# IP Masquerade Agent +# https://docs.cilium.io/en/stable/concepts/networking/masquerading/ +# By default, all packets from a pod destined to an IP address outside of the cilium_native_routing_cidr range are masqueraded +# cilium_ip_masq_agent_enable: false + +### A packet sent from a pod to a destination which belongs to any CIDR from the nonMasqueradeCIDRs is not going to be masqueraded +# cilium_non_masquerade_cidrs: +# - 10.0.0.0/8 +# - 172.16.0.0/12 +# - 192.168.0.0/16 +# - 100.64.0.0/10 +# - 192.0.0.0/24 +# - 192.0.2.0/24 +# - 192.88.99.0/24 +# - 198.18.0.0/15 +# - 198.51.100.0/24 +# - 203.0.113.0/24 +# - 240.0.0.0/4 +### Indicates whether to masquerade traffic to the link local prefix. +### If the masqLinkLocal is not set or set to false, then 169.254.0.0/16 is appended to the non-masquerade CIDRs list. +# cilium_masq_link_local: false +### A time interval at which the agent attempts to reload config from disk +# cilium_ip_masq_resync_interval: 60s + +# Hubble +### Enable Hubble without install +# cilium_enable_hubble: false +### Enable Hubble Metrics +# cilium_enable_hubble_metrics: false +### if cilium_enable_hubble_metrics: true +# cilium_hubble_metrics: {} +# - dns +# - drop +# - tcp +# - flow +# - icmp +# - http +### Enable Hubble install +# cilium_hubble_install: false +### Enable auto generate certs if cilium_hubble_install: true +# cilium_hubble_tls_generate: false + +# IP address management mode for v1.9+. +# https://docs.cilium.io/en/v1.9/concepts/networking/ipam/ +# cilium_ipam_mode: kubernetes + +# Extra arguments for the Cilium agent +# cilium_agent_custom_args: [] + +# For adding and mounting extra volumes to the cilium agent +# cilium_agent_extra_volumes: [] +# cilium_agent_extra_volume_mounts: [] + +# cilium_agent_extra_env_vars: [] + +# cilium_operator_replicas: 2 + +# The address at which the cillium operator bind health check api +# cilium_operator_api_serve_addr: "127.0.0.1:9234" + +## A dictionary of extra config variables to add to cilium-config, formatted like: +## cilium_config_extra_vars: +## var1: "value1" +## var2: "value2" +# cilium_config_extra_vars: {} + +# For adding and mounting extra volumes to the cilium operator +# cilium_operator_extra_volumes: [] +# cilium_operator_extra_volume_mounts: [] + +# Extra arguments for the Cilium Operator +# cilium_operator_custom_args: [] + +# Name of the cluster. Only relevant when building a mesh of clusters. +# cilium_cluster_name: default + +# Make Cilium take ownership over the `/etc/cni/net.d` directory on the node, renaming all non-Cilium CNI configurations to `*.cilium_bak`. +# This ensures no Pods can be scheduled using other CNI plugins during Cilium agent downtime. +# Available for Cilium v1.10 and up. +# cilium_cni_exclusive: true + +# Configure the log file for CNI logging with retention policy of 7 days. +# Disable CNI file logging by setting this field to empty explicitly. +# Available for Cilium v1.12 and up. +# cilium_cni_log_file: "/var/run/cilium/cilium-cni.log" + +# -- Configure cgroup related configuration +# -- Enable auto mount of cgroup2 filesystem. +# When `cilium_cgroup_auto_mount` is enabled, cgroup2 filesystem is mounted at +# `cilium_cgroup_host_root` path on the underlying host and inside the cilium agent pod. +# If users disable `cilium_cgroup_auto_mount`, it's expected that users have mounted +# cgroup2 filesystem at the specified `cilium_cgroup_auto_mount` volume, and then the +# volume will be mounted inside the cilium agent pod at the same path. +# Available for Cilium v1.11 and up +# cilium_cgroup_auto_mount: true +# -- Configure cgroup root where cgroup2 filesystem is mounted on the host +# cilium_cgroup_host_root: "/run/cilium/cgroupv2" + +# Specifies the ratio (0.0-1.0) of total system memory to use for dynamic +# sizing of the TCP CT, non-TCP CT, NAT and policy BPF maps. +# cilium_bpf_map_dynamic_size_ratio: "0.0" + +# -- Enables masquerading of IPv4 traffic leaving the node from endpoints. +# Available for Cilium v1.10 and up +# cilium_enable_ipv4_masquerade: true +# -- Enables masquerading of IPv6 traffic leaving the node from endpoints. +# Available for Cilium v1.10 and up +# cilium_enable_ipv6_masquerade: true + +# -- Enable native IP masquerade support in eBPF +# cilium_enable_bpf_masquerade: false + +# -- Configure whether direct routing mode should route traffic via +# host stack (true) or directly and more efficiently out of BPF (false) if +# the kernel supports it. The latter has the implication that it will also +# bypass netfilter in the host namespace. +# cilium_enable_host_legacy_routing: true + +# -- Enable use of the remote node identity. +# ref: https://docs.cilium.io/en/v1.7/install/upgrade/#configmap-remote-node-identity +# cilium_enable_remote_node_identity: true + +# -- Enable the use of well-known identities. +# cilium_enable_well_known_identities: false + +# cilium_enable_bpf_clock_probe: true + +# -- Whether to enable CNP status updates. +# cilium_disable_cnp_status_updates: true + +# A list of extra rules variables to add to clusterrole for cilium operator, formatted like: +# cilium_clusterrole_rules_operator_extra_vars: +# - apiGroups: +# - '""' +# resources: +# - pods +# verbs: +# - delete +# - apiGroups: +# - '""' +# resources: +# - nodes +# verbs: +# - list +# - watch +# resourceNames: +# - toto +# cilium_clusterrole_rules_operator_extra_vars: [] diff --git a/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-flannel.yml b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-flannel.yml new file mode 100644 index 0000000..64d20a8 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-flannel.yml @@ -0,0 +1,18 @@ +# see roles/network_plugin/flannel/defaults/main.yml + +## interface that should be used for flannel operations +## This is actually an inventory cluster-level item +# flannel_interface: + +## Select interface that should be used for flannel operations by regexp on Name or IP +## This is actually an inventory cluster-level item +## example: select interface with ip from net 10.0.0.0/23 +## single quote and escape backslashes +# flannel_interface_regexp: '10\\.0\\.[0-2]\\.\\d{1,3}' + +# You can choose what type of flannel backend to use: 'vxlan', 'host-gw' or 'wireguard' +# please refer to flannel's docs : https://github.com/coreos/flannel/blob/master/README.md +# flannel_backend_type: "vxlan" +# flannel_vxlan_vni: 1 +# flannel_vxlan_port: 8472 +# flannel_vxlan_direct_routing: false diff --git a/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-kube-ovn.yml b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-kube-ovn.yml new file mode 100644 index 0000000..c241a76 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-kube-ovn.yml @@ -0,0 +1,63 @@ +--- + +# geneve or vlan +kube_ovn_network_type: geneve + +# geneve, vxlan or stt. ATTENTION: some networkpolicy cannot take effect when using vxlan and stt need custom compile ovs kernel module +kube_ovn_tunnel_type: geneve + +## The nic to support container network can be a nic name or a group of regex separated by comma e.g: 'enp6s0f0,eth.*', if empty will use the nic that the default route use. +# kube_ovn_iface: eth1 +## The MTU used by pod iface in overlay networks (default iface MTU - 100) +# kube_ovn_mtu: 1333 + +## Enable hw-offload, disable traffic mirror and set the iface to the physical port. Make sure that there is an IP address bind to the physical port. +kube_ovn_hw_offload: false +# traffic mirror +kube_ovn_traffic_mirror: false + +# kube_ovn_pool_cidr_ipv6: fd85:ee78:d8a6:8607::1:0000/112 +# kube_ovn_default_interface_name: eth0 + +kube_ovn_external_address: 8.8.8.8 +kube_ovn_external_address_ipv6: 2400:3200::1 +kube_ovn_external_dns: alauda.cn + +# kube_ovn_default_gateway: 10.233.64.1,fd85:ee78:d8a6:8607::1:0 +kube_ovn_default_gateway_check: true +kube_ovn_default_logical_gateway: false +# kube_ovn_default_exclude_ips: 10.16.0.1 +kube_ovn_node_switch_cidr: 100.64.0.0/16 +kube_ovn_node_switch_cidr_ipv6: fd00:100:64::/64 + +## vlan config, set default interface name and vlan id +# kube_ovn_default_interface_name: eth0 +kube_ovn_default_vlan_id: 100 +kube_ovn_vlan_name: product + +## pod nic type, support: veth-pair or internal-port +kube_ovn_pod_nic_type: veth_pair + +## Enable load balancer +kube_ovn_enable_lb: true + +## Enable network policy support +kube_ovn_enable_np: true + +## Enable external vpc support +kube_ovn_enable_external_vpc: true + +## Enable checksum +kube_ovn_encap_checksum: true + +## enable ssl +kube_ovn_enable_ssl: false + +## dpdk +kube_ovn_dpdk_enabled: false + +## enable interconnection to an existing IC database server. +kube_ovn_ic_enable: false +kube_ovn_ic_autoroute: true +kube_ovn_ic_dbhost: "127.0.0.1" +kube_ovn_ic_zone: "kubernetes" diff --git a/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-kube-router.yml b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-kube-router.yml new file mode 100644 index 0000000..e4dfcc9 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-kube-router.yml @@ -0,0 +1,64 @@ +# See roles/network_plugin/kube-router//defaults/main.yml + +# Enables Pod Networking -- Advertises and learns the routes to Pods via iBGP +# kube_router_run_router: true + +# Enables Network Policy -- sets up iptables to provide ingress firewall for pods +# kube_router_run_firewall: true + +# Enables Service Proxy -- sets up IPVS for Kubernetes Services +# see docs/kube-router.md "Caveats" section +# kube_router_run_service_proxy: false + +# Add Cluster IP of the service to the RIB so that it gets advertises to the BGP peers. +# kube_router_advertise_cluster_ip: false + +# Add External IP of service to the RIB so that it gets advertised to the BGP peers. +# kube_router_advertise_external_ip: false + +# Add LoadBalancer IP of service status as set by the LB provider to the RIB so that it gets advertised to the BGP peers. +# kube_router_advertise_loadbalancer_ip: false + +# Adjust manifest of kube-router daemonset template with DSR needed changes +# kube_router_enable_dsr: false + +# Array of arbitrary extra arguments to kube-router, see +# https://github.com/cloudnativelabs/kube-router/blob/master/docs/user-guide.md +# kube_router_extra_args: [] + +# ASN number of the cluster, used when communicating with external BGP routers +# kube_router_cluster_asn: ~ + +# ASN numbers of the BGP peer to which cluster nodes will advertise cluster ip and node's pod cidr. +# kube_router_peer_router_asns: ~ + +# The ip address of the external router to which all nodes will peer and advertise the cluster ip and pod cidr's. +# kube_router_peer_router_ips: ~ + +# The remote port of the external BGP to which all nodes will peer. If not set, default BGP port (179) will be used. +# kube_router_peer_router_ports: ~ + +# Setups node CNI to allow hairpin mode, requires node reboots, see +# https://github.com/cloudnativelabs/kube-router/blob/master/docs/user-guide.md#hairpin-mode +# kube_router_support_hairpin_mode: false + +# Select DNS Policy ClusterFirstWithHostNet, ClusterFirst, etc. +# kube_router_dns_policy: ClusterFirstWithHostNet + +# Array of annotations for master +# kube_router_annotations_master: [] + +# Array of annotations for every node +# kube_router_annotations_node: [] + +# Array of common annotations for every node +# kube_router_annotations_all: [] + +# Enables scraping kube-router metrics with Prometheus +# kube_router_enable_metrics: false + +# Path to serve Prometheus metrics on +# kube_router_metrics_path: /metrics + +# Prometheus metrics port to use +# kube_router_metrics_port: 9255 diff --git a/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-macvlan.yml b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-macvlan.yml new file mode 100644 index 0000000..d2534e7 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-macvlan.yml @@ -0,0 +1,6 @@ +--- +# private interface, on a l2-network +macvlan_interface: "eth1" + +# Enable nat in default gateway network interface +enable_nat_default_gateway: true diff --git a/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-weave.yml b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-weave.yml new file mode 100644 index 0000000..269a77c --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/group_vars/k8s_cluster/k8s-net-weave.yml @@ -0,0 +1,64 @@ +# see roles/network_plugin/weave/defaults/main.yml + +# Weave's network password for encryption, if null then no network encryption. +# weave_password: ~ + +# If set to 1, disable checking for new Weave Net versions (default is blank, +# i.e. check is enabled) +# weave_checkpoint_disable: false + +# Soft limit on the number of connections between peers. Defaults to 100. +# weave_conn_limit: 100 + +# Weave Net defaults to enabling hairpin on the bridge side of the veth pair +# for containers attached. If you need to disable hairpin, e.g. your kernel is +# one of those that can panic if hairpin is enabled, then you can disable it by +# setting `HAIRPIN_MODE=false`. +# weave_hairpin_mode: true + +# The range of IP addresses used by Weave Net and the subnet they are placed in +# (CIDR format; default 10.32.0.0/12) +# weave_ipalloc_range: "{{ kube_pods_subnet }}" + +# Set to 0 to disable Network Policy Controller (default is on) +# weave_expect_npc: "{{ enable_network_policy }}" + +# List of addresses of peers in the Kubernetes cluster (default is to fetch the +# list from the api-server) +# weave_kube_peers: ~ + +# Set the initialization mode of the IP Address Manager (defaults to consensus +# amongst the KUBE_PEERS) +# weave_ipalloc_init: ~ + +# Set the IP address used as a gateway from the Weave network to the host +# network - this is useful if you are configuring the addon as a static pod. +# weave_expose_ip: ~ + +# Address and port that the Weave Net daemon will serve Prometheus-style +# metrics on (defaults to 0.0.0.0:6782) +# weave_metrics_addr: ~ + +# Address and port that the Weave Net daemon will serve status requests on +# (defaults to disabled) +# weave_status_addr: ~ + +# Weave Net defaults to 1376 bytes, but you can set a smaller size if your +# underlying network has a tighter limit, or set a larger size for better +# performance if your network supports jumbo frames (e.g. 8916) +# weave_mtu: 1376 + +# Set to 1 to preserve the client source IP address when accessing Service +# annotated with `service.spec.externalTrafficPolicy=Local`. The feature works +# only with Weave IPAM (default). +# weave_no_masq_local: true + +# set to nft to use nftables backend for iptables (default is iptables) +# weave_iptables_backend: iptables + +# Extra variables that passing to launch.sh, useful for enabling seed mode, see +# https://www.weave.works/docs/net/latest/tasks/ipam/ipam/ +# weave_extra_args: ~ + +# Extra variables for weave_npc that passing to launch.sh, useful for change log level, ex --log-level=error +# weave_npc_extra_args: ~ diff --git a/kubespray/inventory/moaroom-cluster/inventory.ini b/kubespray/inventory/moaroom-cluster/inventory.ini new file mode 100644 index 0000000..867c2d3 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/inventory.ini @@ -0,0 +1,42 @@ +# ## Configure 'ip' variable to bind kubernetes services on a +# ## different ip than the default iface +# ## We should set etcd_member_name for etcd cluster. The node that is not a etcd member do not need to set the value, or can set the empty string value. +[all] +control ansible_host=43.202.106.48 +# node1 ansible_host=95.54.0.12 # ip=10.3.0.1 etcd_member_name=etcd1 +# node2 ansible_host=95.54.0.13 # ip=10.3.0.2 etcd_member_name=etcd2 +# node3 ansible_host=95.54.0.14 # ip=10.3.0.3 etcd_member_name=etcd3 +# node4 ansible_host=95.54.0.15 # ip=10.3.0.4 etcd_member_name=etcd4 +# node5 ansible_host=95.54.0.16 # ip=10.3.0.5 etcd_member_name=etcd5 +# node6 ansible_host=95.54.0.17 # ip=10.3.0.6 etcd_member_name=etcd6 + +# ## configure a bastion host if your nodes are not directly reachable +# [bastion] +# bastion ansible_host=x.x.x.x ansible_user=some_user + +[kube_control_plane] +control +# node1 +# node2 +# node3 + +[etcd] +control +# node1 +# node2 +# node3 + +[kube_node] +control +# node2 +# node3 +# node4 +# node5 +# node6 + +[calico_rr] + +[k8s_cluster:children] +kube_control_plane +kube_node +calico_rr diff --git a/kubespray/inventory/moaroom-cluster/patches/kube-controller-manager+merge.yaml b/kubespray/inventory/moaroom-cluster/patches/kube-controller-manager+merge.yaml new file mode 100644 index 0000000..3f0fbbc --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/patches/kube-controller-manager+merge.yaml @@ -0,0 +1,8 @@ +--- +apiVersion: v1 +kind: Pod +metadata: + name: kube-controller-manager + annotations: + prometheus.io/scrape: 'true' + prometheus.io/port: '10257' diff --git a/kubespray/inventory/moaroom-cluster/patches/kube-scheduler+merge.yaml b/kubespray/inventory/moaroom-cluster/patches/kube-scheduler+merge.yaml new file mode 100644 index 0000000..00f4572 --- /dev/null +++ b/kubespray/inventory/moaroom-cluster/patches/kube-scheduler+merge.yaml @@ -0,0 +1,8 @@ +--- +apiVersion: v1 +kind: Pod +metadata: + name: kube-scheduler + annotations: + prometheus.io/scrape: 'true' + prometheus.io/port: '10259' diff --git a/kubespray/inventory/sample/group_vars/all/all.yml b/kubespray/inventory/sample/group_vars/all/all.yml new file mode 100644 index 0000000..b93f1a3 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/all.yml @@ -0,0 +1,139 @@ +--- +## Directory where the binaries will be installed +bin_dir: /usr/local/bin + +## The access_ip variable is used to define how other nodes should access +## the node. This is used in flannel to allow other flannel nodes to see +## this node for example. The access_ip is really useful AWS and Google +## environments where the nodes are accessed remotely by the "public" ip, +## but don't know about that address themselves. +# access_ip: 1.1.1.1 + + +## External LB example config +## apiserver_loadbalancer_domain_name: "elb.some.domain" +# loadbalancer_apiserver: +# address: 1.2.3.4 +# port: 1234 + +## Internal loadbalancers for apiservers +# loadbalancer_apiserver_localhost: true +# valid options are "nginx" or "haproxy" +# loadbalancer_apiserver_type: nginx # valid values "nginx" or "haproxy" + +## Local loadbalancer should use this port +## And must be set port 6443 +loadbalancer_apiserver_port: 6443 + +## If loadbalancer_apiserver_healthcheck_port variable defined, enables proxy liveness check for nginx. +loadbalancer_apiserver_healthcheck_port: 8081 + +### OTHER OPTIONAL VARIABLES + +## By default, Kubespray collects nameservers on the host. It then adds the previously collected nameservers in nameserverentries. +## If true, Kubespray does not include host nameservers in nameserverentries in dns_late stage. However, It uses the nameserver to make sure cluster installed safely in dns_early stage. +## Use this option with caution, you may need to define your dns servers. Otherwise, the outbound queries such as www.google.com may fail. +# disable_host_nameservers: false + +## Upstream dns servers +# upstream_dns_servers: +# - 8.8.8.8 +# - 8.8.4.4 + +## There are some changes specific to the cloud providers +## for instance we need to encapsulate packets with some network plugins +## If set the possible values are either 'gce', 'aws', 'azure', 'openstack', 'vsphere', 'oci', or 'external' +## When openstack is used make sure to source in the openstack credentials +## like you would do when using openstack-client before starting the playbook. +# cloud_provider: + +## When cloud_provider is set to 'external', you can set the cloud controller to deploy +## Supported cloud controllers are: 'openstack', 'vsphere', 'huaweicloud' and 'hcloud' +## When openstack or vsphere are used make sure to source in the required fields +# external_cloud_provider: + +## Set these proxy values in order to update package manager and docker daemon to use proxies and custom CA for https_proxy if needed +# http_proxy: "" +# https_proxy: "" +# https_proxy_cert_file: "" + +## Refer to roles/kubespray-defaults/defaults/main.yml before modifying no_proxy +# no_proxy: "" + +## Some problems may occur when downloading files over https proxy due to ansible bug +## https://github.com/ansible/ansible/issues/32750. Set this variable to False to disable +## SSL validation of get_url module. Note that kubespray will still be performing checksum validation. +# download_validate_certs: False + +## If you need exclude all cluster nodes from proxy and other resources, add other resources here. +# additional_no_proxy: "" + +## If you need to disable proxying of os package repositories but are still behind an http_proxy set +## skip_http_proxy_on_os_packages to true +## This will cause kubespray not to set proxy environment in /etc/yum.conf for centos and in /etc/apt/apt.conf for debian/ubuntu +## Special information for debian/ubuntu - you have to set the no_proxy variable, then apt package will install from your source of wish +# skip_http_proxy_on_os_packages: false + +## Since workers are included in the no_proxy variable by default, docker engine will be restarted on all nodes (all +## pods will restart) when adding or removing workers. To override this behaviour by only including master nodes in the +## no_proxy variable, set below to true: +no_proxy_exclude_workers: false + +## Certificate Management +## This setting determines whether certs are generated via scripts. +## Chose 'none' if you provide your own certificates. +## Option is "script", "none" +# cert_management: script + +## Set to true to allow pre-checks to fail and continue deployment +# ignore_assert_errors: false + +## The read-only port for the Kubelet to serve on with no authentication/authorization. Uncomment to enable. +# kube_read_only_port: 10255 + +## Set true to download and cache container +# download_container: true + +## Deploy container engine +# Set false if you want to deploy container engine manually. +# deploy_container_engine: true + +## Red Hat Enterprise Linux subscription registration +## Add either RHEL subscription Username/Password or Organization ID/Activation Key combination +## Update RHEL subscription purpose usage, role and SLA if necessary +# rh_subscription_username: "" +# rh_subscription_password: "" +# rh_subscription_org_id: "" +# rh_subscription_activation_key: "" +# rh_subscription_usage: "Development" +# rh_subscription_role: "Red Hat Enterprise Server" +# rh_subscription_sla: "Self-Support" + +## Check if access_ip responds to ping. Set false if your firewall blocks ICMP. +# ping_access_ip: true + +# sysctl_file_path to add sysctl conf to +# sysctl_file_path: "/etc/sysctl.d/99-sysctl.conf" + +## Variables for webhook token auth https://kubernetes.io/docs/reference/access-authn-authz/authentication/#webhook-token-authentication +kube_webhook_token_auth: false +kube_webhook_token_auth_url_skip_tls_verify: false +# kube_webhook_token_auth_url: https://... +## base64-encoded string of the webhook's CA certificate +# kube_webhook_token_auth_ca_data: "LS0t..." + +## NTP Settings +# Start the ntpd or chrony service and enable it at system boot. +ntp_enabled: false +ntp_manage_config: false +ntp_servers: + - "0.pool.ntp.org iburst" + - "1.pool.ntp.org iburst" + - "2.pool.ntp.org iburst" + - "3.pool.ntp.org iburst" + +## Used to control no_log attribute +unsafe_show_logs: false + +## If enabled it will allow kubespray to attempt setup even if the distribution is not supported. For unsupported distributions this can lead to unexpected failures in some cases. +allow_unsupported_distribution_setup: false diff --git a/kubespray/inventory/sample/group_vars/all/aws.yml b/kubespray/inventory/sample/group_vars/all/aws.yml new file mode 100644 index 0000000..dab674e --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/aws.yml @@ -0,0 +1,9 @@ +## To use AWS EBS CSI Driver to provision volumes, uncomment the first value +## and configure the parameters below +# aws_ebs_csi_enabled: true +# aws_ebs_csi_enable_volume_scheduling: true +# aws_ebs_csi_enable_volume_snapshot: false +# aws_ebs_csi_enable_volume_resizing: false +# aws_ebs_csi_controller_replicas: 1 +# aws_ebs_csi_plugin_image_tag: latest +# aws_ebs_csi_extra_volume_tags: "Owner=owner,Team=team,Environment=environment' diff --git a/kubespray/inventory/sample/group_vars/all/azure.yml b/kubespray/inventory/sample/group_vars/all/azure.yml new file mode 100644 index 0000000..176b0f1 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/azure.yml @@ -0,0 +1,40 @@ +## When azure is used, you need to also set the following variables. +## see docs/azure.md for details on how to get these values + +# azure_cloud: +# azure_tenant_id: +# azure_subscription_id: +# azure_aad_client_id: +# azure_aad_client_secret: +# azure_resource_group: +# azure_location: +# azure_subnet_name: +# azure_security_group_name: +# azure_security_group_resource_group: +# azure_vnet_name: +# azure_vnet_resource_group: +# azure_route_table_name: +# azure_route_table_resource_group: +# supported values are 'standard' or 'vmss' +# azure_vmtype: standard + +## Azure Disk CSI credentials and parameters +## see docs/azure-csi.md for details on how to get these values + +# azure_csi_tenant_id: +# azure_csi_subscription_id: +# azure_csi_aad_client_id: +# azure_csi_aad_client_secret: +# azure_csi_location: +# azure_csi_resource_group: +# azure_csi_vnet_name: +# azure_csi_vnet_resource_group: +# azure_csi_subnet_name: +# azure_csi_security_group_name: +# azure_csi_use_instance_metadata: +# azure_csi_tags: "Owner=owner,Team=team,Environment=environment' + +## To enable Azure Disk CSI, uncomment below +# azure_csi_enabled: true +# azure_csi_controller_replicas: 1 +# azure_csi_plugin_image_tag: latest diff --git a/kubespray/inventory/sample/group_vars/all/containerd.yml b/kubespray/inventory/sample/group_vars/all/containerd.yml new file mode 100644 index 0000000..1888b24 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/containerd.yml @@ -0,0 +1,46 @@ +--- +# Please see roles/container-engine/containerd/defaults/main.yml for more configuration options + +# containerd_storage_dir: "/var/lib/containerd" +# containerd_state_dir: "/run/containerd" +# containerd_oom_score: 0 + +# containerd_default_runtime: "runc" +# containerd_snapshotter: "native" + +# containerd_runc_runtime: +# name: runc +# type: "io.containerd.runc.v2" +# engine: "" +# root: "" + +# containerd_additional_runtimes: +# Example for Kata Containers as additional runtime: +# - name: kata +# type: "io.containerd.kata.v2" +# engine: "" +# root: "" + +# containerd_grpc_max_recv_message_size: 16777216 +# containerd_grpc_max_send_message_size: 16777216 + +# containerd_debug_level: "info" + +# containerd_metrics_address: "" + +# containerd_metrics_grpc_histogram: false + +# Registries defined within containerd. +# containerd_registries_mirrors: +# - prefix: docker.io +# mirrors: +# - host: https://registry-1.docker.io +# capabilities: ["pull", "resolve"] +# skip_verify: false + +# containerd_max_container_log_line_size: -1 + +# containerd_registry_auth: +# - registry: 10.0.0.2:5000 +# username: user +# password: pass diff --git a/kubespray/inventory/sample/group_vars/all/coreos.yml b/kubespray/inventory/sample/group_vars/all/coreos.yml new file mode 100644 index 0000000..22c2166 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/coreos.yml @@ -0,0 +1,2 @@ +## Does coreos need auto upgrade, default is true +# coreos_auto_upgrade: true diff --git a/kubespray/inventory/sample/group_vars/all/cri-o.yml b/kubespray/inventory/sample/group_vars/all/cri-o.yml new file mode 100644 index 0000000..3e6e4ee --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/cri-o.yml @@ -0,0 +1,6 @@ +# crio_insecure_registries: +# - 10.0.0.2:5000 +# crio_registry_auth: +# - registry: 10.0.0.2:5000 +# username: user +# password: pass diff --git a/kubespray/inventory/sample/group_vars/all/docker.yml b/kubespray/inventory/sample/group_vars/all/docker.yml new file mode 100644 index 0000000..4e968c3 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/docker.yml @@ -0,0 +1,59 @@ +--- +## Uncomment this if you want to force overlay/overlay2 as docker storage driver +## Please note that overlay2 is only supported on newer kernels +# docker_storage_options: -s overlay2 + +## Enable docker_container_storage_setup, it will configure devicemapper driver on Centos7 or RedHat7. +docker_container_storage_setup: false + +## It must be define a disk path for docker_container_storage_setup_devs. +## Otherwise docker-storage-setup will be executed incorrectly. +# docker_container_storage_setup_devs: /dev/vdb + +## Uncomment this if you want to change the Docker Cgroup driver (native.cgroupdriver) +## Valid options are systemd or cgroupfs, default is systemd +# docker_cgroup_driver: systemd + +## Only set this if you have more than 3 nameservers: +## If true Kubespray will only use the first 3, otherwise it will fail +docker_dns_servers_strict: false + +# Path used to store Docker data +docker_daemon_graph: "/var/lib/docker" + +## Used to set docker daemon iptables options to true +docker_iptables_enabled: "false" + +# Docker log options +# Rotate container stderr/stdout logs at 50m and keep last 5 +docker_log_opts: "--log-opt max-size=50m --log-opt max-file=5" + +# define docker bin_dir +docker_bin_dir: "/usr/bin" + +# keep docker packages after installation; speeds up repeated ansible provisioning runs when '1' +# kubespray deletes the docker package on each run, so caching the package makes sense +docker_rpm_keepcache: 1 + +## An obvious use case is allowing insecure-registry access to self hosted registries. +## Can be ipaddress and domain_name. +## example define 172.19.16.11 or mirror.registry.io +# docker_insecure_registries: +# - mirror.registry.io +# - 172.19.16.11 + +## Add other registry,example China registry mirror. +# docker_registry_mirrors: +# - https://registry.docker-cn.com +# - https://mirror.aliyuncs.com + +## If non-empty will override default system MountFlags value. +## This option takes a mount propagation flag: shared, slave +## or private, which control whether mounts in the file system +## namespace set up for docker will receive or propagate mounts +## and unmounts. Leave empty for system default +# docker_mount_flags: + +## A string of extra options to pass to the docker daemon. +## This string should be exactly as you wish it to appear. +# docker_options: "" diff --git a/kubespray/inventory/sample/group_vars/all/etcd.yml b/kubespray/inventory/sample/group_vars/all/etcd.yml new file mode 100644 index 0000000..39600c3 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/etcd.yml @@ -0,0 +1,16 @@ +--- +## Directory where etcd data stored +etcd_data_dir: /var/lib/etcd + +## Container runtime +## docker for docker, crio for cri-o and containerd for containerd. +## Additionally you can set this to kubeadm if you want to install etcd using kubeadm +## Kubeadm etcd deployment is experimental and only available for new deployments +## If this is not set, container manager will be inherited from the Kubespray defaults +## and not from k8s_cluster/k8s-cluster.yml, which might not be what you want. +## Also this makes possible to use different container manager for etcd nodes. +# container_manager: containerd + +## Settings for etcd deployment type +# Set this to docker if you are using container_manager: docker +etcd_deployment_type: host diff --git a/kubespray/inventory/sample/group_vars/all/gcp.yml b/kubespray/inventory/sample/group_vars/all/gcp.yml new file mode 100644 index 0000000..49eb5c0 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/gcp.yml @@ -0,0 +1,10 @@ +## GCP compute Persistent Disk CSI Driver credentials and parameters +## See docs/gcp-pd-csi.md for information about the implementation + +## Specify the path to the file containing the service account credentials +# gcp_pd_csi_sa_cred_file: "/my/safe/credentials/directory/cloud-sa.json" + +## To enable GCP Persistent Disk CSI driver, uncomment below +# gcp_pd_csi_enabled: true +# gcp_pd_csi_controller_replicas: 1 +# gcp_pd_csi_driver_image_tag: "v0.7.0-gke.0" diff --git a/kubespray/inventory/sample/group_vars/all/hcloud.yml b/kubespray/inventory/sample/group_vars/all/hcloud.yml new file mode 100644 index 0000000..d4ed65c --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/hcloud.yml @@ -0,0 +1,22 @@ +## Values for the external Hcloud Cloud Controller +# external_hcloud_cloud: +# hcloud_api_token: "" +# token_secret_name: hcloud +# with_networks: false # Use the hcloud controller-manager with networks support https://github.com/hetznercloud/hcloud-cloud-controller-manager#networks-support +# network_name: # network name/ID: If you manage the network yourself it might still be required to let the CCM know about private networks +# service_account_name: cloud-controller-manager +# +# controller_image_tag: "latest" +# ## A dictionary of extra arguments to add to the openstack cloud controller manager daemonset +# ## Format: +# ## external_hcloud_cloud.controller_extra_args: +# ## arg1: "value1" +# ## arg2: "value2" +# controller_extra_args: {} +# +# load_balancers_location: # mutually exclusive with load_balancers_network_zone +# load_balancers_network_zone: +# load_balancers_disable_private_ingress: # set to true if using IPVS based plugins https://github.com/hetznercloud/hcloud-cloud-controller-manager/blob/main/docs/load_balancers.md#sample-service-with-networks +# load_balancers_use_private_ip: # set to true if using private networks +# load_balancers_enabled: +# network_routes_enabled: diff --git a/kubespray/inventory/sample/group_vars/all/huaweicloud.yml b/kubespray/inventory/sample/group_vars/all/huaweicloud.yml new file mode 100644 index 0000000..20c7202 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/huaweicloud.yml @@ -0,0 +1,17 @@ +## Values for the external Huawei Cloud Controller +# external_huaweicloud_lbaas_subnet_id: "Neutron subnet ID to create LBaaS VIP" +# external_huaweicloud_lbaas_network_id: "Neutron network ID to create LBaaS VIP" + +## Credentials to authenticate against Keystone API +## All of them are required Per default these values will be +## read from the environment. +# external_huaweicloud_auth_url: "{{ lookup('env','OS_AUTH_URL') }}" +# external_huaweicloud_access_key: "{{ lookup('env','OS_ACCESS_KEY') }}" +# external_huaweicloud_secret_key: "{{ lookup('env','OS_SECRET_KEY') }}" +# external_huaweicloud_region: "{{ lookup('env','OS_REGION_NAME') }}" +# external_huaweicloud_project_id: "{{ lookup('env','OS_TENANT_ID')| default(lookup('env','OS_PROJECT_ID'),true) }}" +# external_huaweicloud_cloud: "{{ lookup('env','OS_CLOUD') }}" + +## The repo and tag of the external Huawei Cloud Controller image +# external_huawei_cloud_controller_image_repo: "swr.ap-southeast-1.myhuaweicloud.com" +# external_huawei_cloud_controller_image_tag: "v0.26.3" diff --git a/kubespray/inventory/sample/group_vars/all/oci.yml b/kubespray/inventory/sample/group_vars/all/oci.yml new file mode 100644 index 0000000..541d0e6 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/oci.yml @@ -0,0 +1,28 @@ +## When Oracle Cloud Infrastructure is used, set these variables +# oci_private_key: +# oci_region_id: +# oci_tenancy_id: +# oci_user_id: +# oci_user_fingerprint: +# oci_compartment_id: +# oci_vnc_id: +# oci_subnet1_id: +# oci_subnet2_id: +## Override these default/optional behaviors if you wish +# oci_security_list_management: All +## If you would like the controller to manage specific lists per subnet. This is a mapping of subnet ocids to security list ocids. Below are examples. +# oci_security_lists: +# ocid1.subnet.oc1.phx.aaaaaaaasa53hlkzk6nzksqfccegk2qnkxmphkblst3riclzs4rhwg7rg57q: ocid1.securitylist.oc1.iad.aaaaaaaaqti5jsfvyw6ejahh7r4okb2xbtuiuguswhs746mtahn72r7adt7q +# ocid1.subnet.oc1.phx.aaaaaaaahuxrgvs65iwdz7ekwgg3l5gyah7ww5klkwjcso74u3e4i64hvtvq: ocid1.securitylist.oc1.iad.aaaaaaaaqti5jsfvyw6ejahh7r4okb2xbtuiuguswhs746mtahn72r7adt7q +## If oci_use_instance_principals is true, you do not need to set the region, tenancy, user, key, passphrase, or fingerprint +# oci_use_instance_principals: false +# oci_cloud_controller_version: 0.6.0 +## If you would like to control OCI query rate limits for the controller +# oci_rate_limit: +# rate_limit_qps_read: +# rate_limit_qps_write: +# rate_limit_bucket_read: +# rate_limit_bucket_write: +## Other optional variables +# oci_cloud_controller_pull_source: (default iad.ocir.io/oracle/cloud-provider-oci) +# oci_cloud_controller_pull_secret: (name of pull secret to use if you define your own mirror above) diff --git a/kubespray/inventory/sample/group_vars/all/offline.yml b/kubespray/inventory/sample/group_vars/all/offline.yml new file mode 100644 index 0000000..7fba57e --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/offline.yml @@ -0,0 +1,106 @@ +--- +## Global Offline settings +### Private Container Image Registry +# registry_host: "myprivateregisry.com" +# files_repo: "http://myprivatehttpd" +### If using CentOS, RedHat, AlmaLinux or Fedora +# yum_repo: "http://myinternalyumrepo" +### If using Debian +# debian_repo: "http://myinternaldebianrepo" +### If using Ubuntu +# ubuntu_repo: "http://myinternalubunturepo" + +## Container Registry overrides +# kube_image_repo: "{{ registry_host }}" +# gcr_image_repo: "{{ registry_host }}" +# github_image_repo: "{{ registry_host }}" +# docker_image_repo: "{{ registry_host }}" +# quay_image_repo: "{{ registry_host }}" + +## Kubernetes components +# kubeadm_download_url: "{{ files_repo }}/dl.k8s.io/release/{{ kubeadm_version }}/bin/linux/{{ image_arch }}/kubeadm" +# kubectl_download_url: "{{ files_repo }}/dl.k8s.io/release/{{ kube_version }}/bin/linux/{{ image_arch }}/kubectl" +# kubelet_download_url: "{{ files_repo }}/dl.k8s.io/release/{{ kube_version }}/bin/linux/{{ image_arch }}/kubelet" + +## CNI Plugins +# cni_download_url: "{{ files_repo }}/github.com/containernetworking/plugins/releases/download/{{ cni_version }}/cni-plugins-linux-{{ image_arch }}-{{ cni_version }}.tgz" + +## cri-tools +# crictl_download_url: "{{ files_repo }}/github.com/kubernetes-sigs/cri-tools/releases/download/{{ crictl_version }}/crictl-{{ crictl_version }}-{{ ansible_system | lower }}-{{ image_arch }}.tar.gz" + +## [Optional] etcd: only if you use etcd_deployment=host +# etcd_download_url: "{{ files_repo }}/github.com/etcd-io/etcd/releases/download/{{ etcd_version }}/etcd-{{ etcd_version }}-linux-{{ image_arch }}.tar.gz" + +# [Optional] Calico: If using Calico network plugin +# calicoctl_download_url: "{{ files_repo }}/github.com/projectcalico/calico/releases/download/{{ calico_ctl_version }}/calicoctl-linux-{{ image_arch }}" +# [Optional] Calico with kdd: If using Calico network plugin with kdd datastore +# calico_crds_download_url: "{{ files_repo }}/github.com/projectcalico/calico/archive/{{ calico_version }}.tar.gz" + +# [Optional] Cilium: If using Cilium network plugin +# ciliumcli_download_url: "{{ files_repo }}/github.com/cilium/cilium-cli/releases/download/{{ cilium_cli_version }}/cilium-linux-{{ image_arch }}.tar.gz" + +# [Optional] helm: only if you set helm_enabled: true +# helm_download_url: "{{ files_repo }}/get.helm.sh/helm-{{ helm_version }}-linux-{{ image_arch }}.tar.gz" + +# [Optional] crun: only if you set crun_enabled: true +# crun_download_url: "{{ files_repo }}/github.com/containers/crun/releases/download/{{ crun_version }}/crun-{{ crun_version }}-linux-{{ image_arch }}" + +# [Optional] kata: only if you set kata_containers_enabled: true +# kata_containers_download_url: "{{ files_repo }}/github.com/kata-containers/kata-containers/releases/download/{{ kata_containers_version }}/kata-static-{{ kata_containers_version }}-{{ ansible_architecture }}.tar.xz" + +# [Optional] cri-dockerd: only if you set container_manager: docker +# cri_dockerd_download_url: "{{ files_repo }}/github.com/Mirantis/cri-dockerd/releases/download/v{{ cri_dockerd_version }}/cri-dockerd-{{ cri_dockerd_version }}.{{ image_arch }}.tgz" + +# [Optional] runc: if you set container_manager to containerd or crio +# runc_download_url: "{{ files_repo }}/github.com/opencontainers/runc/releases/download/{{ runc_version }}/runc.{{ image_arch }}" + +# [Optional] cri-o: only if you set container_manager: crio +# crio_download_base: "download.opensuse.org/repositories/devel:kubic:libcontainers:stable" +# crio_download_crio: "http://{{ crio_download_base }}:/cri-o:/" +# crio_download_url: "{{ files_repo }}/storage.googleapis.com/cri-o/artifacts/cri-o.{{ image_arch }}.{{ crio_version }}.tar.gz" +# skopeo_download_url: "{{ files_repo }}/github.com/lework/skopeo-binary/releases/download/{{ skopeo_version }}/skopeo-linux-{{ image_arch }}" + +# [Optional] containerd: only if you set container_runtime: containerd +# containerd_download_url: "{{ files_repo }}/github.com/containerd/containerd/releases/download/v{{ containerd_version }}/containerd-{{ containerd_version }}-linux-{{ image_arch }}.tar.gz" +# nerdctl_download_url: "{{ files_repo }}/github.com/containerd/nerdctl/releases/download/v{{ nerdctl_version }}/nerdctl-{{ nerdctl_version }}-{{ ansible_system | lower }}-{{ image_arch }}.tar.gz" + +# [Optional] runsc,containerd-shim-runsc: only if you set gvisor_enabled: true +# gvisor_runsc_download_url: "{{ files_repo }}/storage.googleapis.com/gvisor/releases/release/{{ gvisor_version }}/{{ ansible_architecture }}/runsc" +# gvisor_containerd_shim_runsc_download_url: "{{ files_repo }}/storage.googleapis.com/gvisor/releases/release/{{ gvisor_version }}/{{ ansible_architecture }}/containerd-shim-runsc-v1" + +# [Optional] Krew: only if you set krew_enabled: true +# krew_download_url: "{{ files_repo }}/github.com/kubernetes-sigs/krew/releases/download/{{ krew_version }}/krew-{{ host_os }}_{{ image_arch }}.tar.gz" + +## CentOS/Redhat/AlmaLinux +### For EL7, base and extras repo must be available, for EL8, baseos and appstream +### By default we enable those repo automatically +# rhel_enable_repos: false +### Docker / Containerd +# docker_rh_repo_base_url: "{{ yum_repo }}/docker-ce/$releasever/$basearch" +# docker_rh_repo_gpgkey: "{{ yum_repo }}/docker-ce/gpg" + +## Fedora +### Docker +# docker_fedora_repo_base_url: "{{ yum_repo }}/docker-ce/{{ ansible_distribution_major_version }}/{{ ansible_architecture }}" +# docker_fedora_repo_gpgkey: "{{ yum_repo }}/docker-ce/gpg" +### Containerd +# containerd_fedora_repo_base_url: "{{ yum_repo }}/containerd" +# containerd_fedora_repo_gpgkey: "{{ yum_repo }}/docker-ce/gpg" + +## Debian +### Docker +# docker_debian_repo_base_url: "{{ debian_repo }}/docker-ce" +# docker_debian_repo_gpgkey: "{{ debian_repo }}/docker-ce/gpg" +### Containerd +# containerd_debian_repo_base_url: "{{ debian_repo }}/containerd" +# containerd_debian_repo_gpgkey: "{{ debian_repo }}/containerd/gpg" +# containerd_debian_repo_repokey: 'YOURREPOKEY' + +## Ubuntu +### Docker +# docker_ubuntu_repo_base_url: "{{ ubuntu_repo }}/docker-ce" +# docker_ubuntu_repo_gpgkey: "{{ ubuntu_repo }}/docker-ce/gpg" +### Containerd +# containerd_ubuntu_repo_base_url: "{{ ubuntu_repo }}/containerd" +# containerd_ubuntu_repo_gpgkey: "{{ ubuntu_repo }}/containerd/gpg" +# containerd_ubuntu_repo_repokey: 'YOURREPOKEY' diff --git a/kubespray/inventory/sample/group_vars/all/openstack.yml b/kubespray/inventory/sample/group_vars/all/openstack.yml new file mode 100644 index 0000000..0fec79a --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/openstack.yml @@ -0,0 +1,50 @@ +## When OpenStack is used, Cinder version can be explicitly specified if autodetection fails (Fixed in 1.9: https://github.com/kubernetes/kubernetes/issues/50461) +# openstack_blockstorage_version: "v1/v2/auto (default)" +# openstack_blockstorage_ignore_volume_az: yes +## When OpenStack is used, if LBaaSv2 is available you can enable it with the following 2 variables. +# openstack_lbaas_enabled: True +# openstack_lbaas_subnet_id: "Neutron subnet ID (not network ID) to create LBaaS VIP" +## To enable automatic floating ip provisioning, specify a subnet. +# openstack_lbaas_floating_network_id: "Neutron network ID (not subnet ID) to get floating IP from, disabled by default" +## Override default LBaaS behavior +# openstack_lbaas_use_octavia: False +# openstack_lbaas_method: "ROUND_ROBIN" +# openstack_lbaas_provider: "haproxy" +# openstack_lbaas_create_monitor: "yes" +# openstack_lbaas_monitor_delay: "1m" +# openstack_lbaas_monitor_timeout: "30s" +# openstack_lbaas_monitor_max_retries: "3" + +## Values for the external OpenStack Cloud Controller +# external_openstack_lbaas_enabled: true +# external_openstack_lbaas_floating_network_id: "Neutron network ID to get floating IP from" +# external_openstack_lbaas_floating_subnet_id: "Neutron subnet ID to get floating IP from" +# external_openstack_lbaas_method: ROUND_ROBIN +# external_openstack_lbaas_provider: amphora +# external_openstack_lbaas_subnet_id: "Neutron subnet ID to create LBaaS VIP" +# external_openstack_lbaas_network_id: "Neutron network ID to create LBaaS VIP" +# external_openstack_lbaas_manage_security_groups: false +# external_openstack_lbaas_create_monitor: false +# external_openstack_lbaas_monitor_delay: 5 +# external_openstack_lbaas_monitor_max_retries: 1 +# external_openstack_lbaas_monitor_timeout: 3 +# external_openstack_lbaas_internal_lb: false +# external_openstack_network_ipv6_disabled: false +# external_openstack_network_internal_networks: [] +# external_openstack_network_public_networks: [] +# external_openstack_metadata_search_order: "configDrive,metadataService" + +## Application credentials to authenticate against Keystone API +## Those settings will take precedence over username and password that might be set your environment +## All of them are required +# external_openstack_application_credential_name: +# external_openstack_application_credential_id: +# external_openstack_application_credential_secret: + +## The tag of the external OpenStack Cloud Controller image +# external_openstack_cloud_controller_image_tag: "latest" + +## To use Cinder CSI plugin to provision volumes set this value to true +## Make sure to source in the openstack credentials +# cinder_csi_enabled: true +# cinder_csi_controller_replicas: 1 diff --git a/kubespray/inventory/sample/group_vars/all/upcloud.yml b/kubespray/inventory/sample/group_vars/all/upcloud.yml new file mode 100644 index 0000000..f05435d --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/upcloud.yml @@ -0,0 +1,24 @@ +## Repo for UpClouds csi-driver: https://github.com/UpCloudLtd/upcloud-csi +## To use UpClouds CSI plugin to provision volumes set this value to true +## Remember to set UPCLOUD_USERNAME and UPCLOUD_PASSWORD +# upcloud_csi_enabled: true +# upcloud_csi_controller_replicas: 1 +## Override used image tags +# upcloud_csi_provisioner_image_tag: "v3.1.0" +# upcloud_csi_attacher_image_tag: "v3.4.0" +# upcloud_csi_resizer_image_tag: "v1.4.0" +# upcloud_csi_plugin_image_tag: "v0.3.3" +# upcloud_csi_node_image_tag: "v2.5.0" +# upcloud_tolerations: [] +## Storage class options +# storage_classes: +# - name: standard +# is_default: true +# expand_persistent_volumes: true +# parameters: +# tier: maxiops +# - name: hdd +# is_default: false +# expand_persistent_volumes: true +# parameters: +# tier: hdd diff --git a/kubespray/inventory/sample/group_vars/all/vsphere.yml b/kubespray/inventory/sample/group_vars/all/vsphere.yml new file mode 100644 index 0000000..af3cfbe --- /dev/null +++ b/kubespray/inventory/sample/group_vars/all/vsphere.yml @@ -0,0 +1,32 @@ +## Values for the external vSphere Cloud Provider +# external_vsphere_vcenter_ip: "myvcenter.domain.com" +# external_vsphere_vcenter_port: "443" +# external_vsphere_insecure: "true" +# external_vsphere_user: "administrator@vsphere.local" # Can also be set via the `VSPHERE_USER` environment variable +# external_vsphere_password: "K8s_admin" # Can also be set via the `VSPHERE_PASSWORD` environment variable +# external_vsphere_datacenter: "DATACENTER_name" +# external_vsphere_kubernetes_cluster_id: "kubernetes-cluster-id" + +## Vsphere version where located VMs +# external_vsphere_version: "6.7u3" + +## Tags for the external vSphere Cloud Provider images +## gcr.io/cloud-provider-vsphere/cpi/release/manager +# external_vsphere_cloud_controller_image_tag: "latest" +## gcr.io/cloud-provider-vsphere/csi/release/syncer +# vsphere_syncer_image_tag: "v2.5.1" +## registry.k8s.io/sig-storage/csi-attacher +# vsphere_csi_attacher_image_tag: "v3.4.0" +## gcr.io/cloud-provider-vsphere/csi/release/driver +# vsphere_csi_controller: "v2.5.1" +## registry.k8s.io/sig-storage/livenessprobe +# vsphere_csi_liveness_probe_image_tag: "v2.6.0" +## registry.k8s.io/sig-storage/csi-provisioner +# vsphere_csi_provisioner_image_tag: "v3.1.0" +## registry.k8s.io/sig-storage/csi-resizer +## makes sense only for vSphere version >=7.0 +# vsphere_csi_resizer_tag: "v1.3.0" + +## To use vSphere CSI plugin to provision volumes set this value to true +# vsphere_csi_enabled: true +# vsphere_csi_controller_replicas: 1 diff --git a/kubespray/inventory/sample/group_vars/etcd.yml b/kubespray/inventory/sample/group_vars/etcd.yml new file mode 100644 index 0000000..f07c720 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/etcd.yml @@ -0,0 +1,26 @@ +--- +## Etcd auto compaction retention for mvcc key value store in hour +# etcd_compaction_retention: 0 + +## Set level of detail for etcd exported metrics, specify 'extensive' to include histogram metrics. +# etcd_metrics: basic + +## Etcd is restricted by default to 512M on systems under 4GB RAM, 512MB is not enough for much more than testing. +## Set this if your etcd nodes have less than 4GB but you want more RAM for etcd. Set to 0 for unrestricted RAM. +## This value is only relevant when deploying etcd with `etcd_deployment_type: docker` +# etcd_memory_limit: "512M" + +## Etcd has a default of 2G for its space quota. If you put a value in etcd_memory_limit which is less than +## etcd_quota_backend_bytes, you may encounter out of memory terminations of the etcd cluster. Please check +## etcd documentation for more information. +# 8G is a suggested maximum size for normal environments and etcd warns at startup if the configured value exceeds it. +# etcd_quota_backend_bytes: "2147483648" + +# Maximum client request size in bytes the server will accept. +# etcd is designed to handle small key value pairs typical for metadata. +# Larger requests will work, but may increase the latency of other requests +# etcd_max_request_bytes: "1572864" + +### ETCD: disable peer client cert authentication. +# This affects ETCD_PEER_CLIENT_CERT_AUTH variable +# etcd_peer_client_auth: true diff --git a/kubespray/inventory/sample/group_vars/k8s_cluster/addons.yml b/kubespray/inventory/sample/group_vars/k8s_cluster/addons.yml new file mode 100644 index 0000000..38bc8ca --- /dev/null +++ b/kubespray/inventory/sample/group_vars/k8s_cluster/addons.yml @@ -0,0 +1,261 @@ +--- +# Kubernetes dashboard +# RBAC required. see docs/getting-started.md for access details. +# dashboard_enabled: false + +# Helm deployment +helm_enabled: false + +# Registry deployment +registry_enabled: false +# registry_namespace: kube-system +# registry_storage_class: "" +# registry_disk_size: "10Gi" + +# Metrics Server deployment +metrics_server_enabled: false +# metrics_server_container_port: 10250 +# metrics_server_kubelet_insecure_tls: true +# metrics_server_metric_resolution: 15s +# metrics_server_kubelet_preferred_address_types: "InternalIP,ExternalIP,Hostname" +# metrics_server_host_network: false +# metrics_server_replicas: 1 + +# Rancher Local Path Provisioner +local_path_provisioner_enabled: false +# local_path_provisioner_namespace: "local-path-storage" +# local_path_provisioner_storage_class: "local-path" +# local_path_provisioner_reclaim_policy: Delete +# local_path_provisioner_claim_root: /opt/local-path-provisioner/ +# local_path_provisioner_debug: false +# local_path_provisioner_image_repo: "rancher/local-path-provisioner" +# local_path_provisioner_image_tag: "v0.0.24" +# local_path_provisioner_helper_image_repo: "busybox" +# local_path_provisioner_helper_image_tag: "latest" + +# Local volume provisioner deployment +local_volume_provisioner_enabled: false +# local_volume_provisioner_namespace: kube-system +# local_volume_provisioner_nodelabels: +# - kubernetes.io/hostname +# - topology.kubernetes.io/region +# - topology.kubernetes.io/zone +# local_volume_provisioner_storage_classes: +# local-storage: +# host_dir: /mnt/disks +# mount_dir: /mnt/disks +# volume_mode: Filesystem +# fs_type: ext4 +# fast-disks: +# host_dir: /mnt/fast-disks +# mount_dir: /mnt/fast-disks +# block_cleaner_command: +# - "/scripts/shred.sh" +# - "2" +# volume_mode: Filesystem +# fs_type: ext4 +# local_volume_provisioner_tolerations: +# - effect: NoSchedule +# operator: Exists + +# CSI Volume Snapshot Controller deployment, set this to true if your CSI is able to manage snapshots +# currently, setting cinder_csi_enabled=true would automatically enable the snapshot controller +# Longhorn is an extenal CSI that would also require setting this to true but it is not included in kubespray +# csi_snapshot_controller_enabled: false +# csi snapshot namespace +# snapshot_controller_namespace: kube-system + +# CephFS provisioner deployment +cephfs_provisioner_enabled: false +# cephfs_provisioner_namespace: "cephfs-provisioner" +# cephfs_provisioner_cluster: ceph +# cephfs_provisioner_monitors: "172.24.0.1:6789,172.24.0.2:6789,172.24.0.3:6789" +# cephfs_provisioner_admin_id: admin +# cephfs_provisioner_secret: secret +# cephfs_provisioner_storage_class: cephfs +# cephfs_provisioner_reclaim_policy: Delete +# cephfs_provisioner_claim_root: /volumes +# cephfs_provisioner_deterministic_names: true + +# RBD provisioner deployment +rbd_provisioner_enabled: false +# rbd_provisioner_namespace: rbd-provisioner +# rbd_provisioner_replicas: 2 +# rbd_provisioner_monitors: "172.24.0.1:6789,172.24.0.2:6789,172.24.0.3:6789" +# rbd_provisioner_pool: kube +# rbd_provisioner_admin_id: admin +# rbd_provisioner_secret_name: ceph-secret-admin +# rbd_provisioner_secret: ceph-key-admin +# rbd_provisioner_user_id: kube +# rbd_provisioner_user_secret_name: ceph-secret-user +# rbd_provisioner_user_secret: ceph-key-user +# rbd_provisioner_user_secret_namespace: rbd-provisioner +# rbd_provisioner_fs_type: ext4 +# rbd_provisioner_image_format: "2" +# rbd_provisioner_image_features: layering +# rbd_provisioner_storage_class: rbd +# rbd_provisioner_reclaim_policy: Delete + +# Nginx ingress controller deployment +ingress_nginx_enabled: false +# ingress_nginx_host_network: false +ingress_publish_status_address: "" +# ingress_nginx_nodeselector: +# kubernetes.io/os: "linux" +# ingress_nginx_tolerations: +# - key: "node-role.kubernetes.io/master" +# operator: "Equal" +# value: "" +# effect: "NoSchedule" +# - key: "node-role.kubernetes.io/control-plane" +# operator: "Equal" +# value: "" +# effect: "NoSchedule" +# ingress_nginx_namespace: "ingress-nginx" +# ingress_nginx_insecure_port: 80 +# ingress_nginx_secure_port: 443 +# ingress_nginx_configmap: +# map-hash-bucket-size: "128" +# ssl-protocols: "TLSv1.2 TLSv1.3" +# ingress_nginx_configmap_tcp_services: +# 9000: "default/example-go:8080" +# ingress_nginx_configmap_udp_services: +# 53: "kube-system/coredns:53" +# ingress_nginx_extra_args: +# - --default-ssl-certificate=default/foo-tls +# ingress_nginx_termination_grace_period_seconds: 300 +# ingress_nginx_class: nginx +# ingress_nginx_without_class: true +# ingress_nginx_default: false + +# ALB ingress controller deployment +ingress_alb_enabled: false +# alb_ingress_aws_region: "us-east-1" +# alb_ingress_restrict_scheme: "false" +# Enables logging on all outbound requests sent to the AWS API. +# If logging is desired, set to true. +# alb_ingress_aws_debug: "false" + +# Cert manager deployment +cert_manager_enabled: false +# cert_manager_namespace: "cert-manager" +# cert_manager_tolerations: +# - key: node-role.kubernetes.io/master +# effect: NoSchedule +# - key: node-role.kubernetes.io/control-plane +# effect: NoSchedule +# cert_manager_affinity: +# nodeAffinity: +# preferredDuringSchedulingIgnoredDuringExecution: +# - weight: 100 +# preference: +# matchExpressions: +# - key: node-role.kubernetes.io/control-plane +# operator: In +# values: +# - "" +# cert_manager_nodeselector: +# kubernetes.io/os: "linux" + +# cert_manager_trusted_internal_ca: | +# -----BEGIN CERTIFICATE----- +# [REPLACE with your CA certificate] +# -----END CERTIFICATE----- +# cert_manager_leader_election_namespace: kube-system + +# cert_manager_dns_policy: "ClusterFirst" +# cert_manager_dns_config: +# nameservers: +# - "1.1.1.1" +# - "8.8.8.8" + +# cert_manager_controller_extra_args: +# - "--dns01-recursive-nameservers-only=true" +# - "--dns01-recursive-nameservers=1.1.1.1:53,8.8.8.8:53" + +# MetalLB deployment +metallb_enabled: false +metallb_speaker_enabled: "{{ metallb_enabled }}" +# metallb_speaker_nodeselector: +# kubernetes.io/os: "linux" +# metallb_controller_nodeselector: +# kubernetes.io/os: "linux" +# metallb_speaker_tolerations: +# - key: "node-role.kubernetes.io/master" +# operator: "Equal" +# value: "" +# effect: "NoSchedule" +# - key: "node-role.kubernetes.io/control-plane" +# operator: "Equal" +# value: "" +# effect: "NoSchedule" +# metallb_controller_tolerations: +# - key: "node-role.kubernetes.io/master" +# operator: "Equal" +# value: "" +# effect: "NoSchedule" +# - key: "node-role.kubernetes.io/control-plane" +# operator: "Equal" +# value: "" +# effect: "NoSchedule" +# metallb_version: v0.13.9 +# metallb_protocol: "layer2" +# metallb_port: "7472" +# metallb_memberlist_port: "7946" +# metallb_config: +# address_pools: +# primary: +# ip_range: +# - 10.5.0.0/16 +# auto_assign: true +# pool1: +# ip_range: +# - 10.6.0.0/16 +# auto_assign: true +# pool2: +# ip_range: +# - 10.10.0.0/16 +# auto_assign: true +# layer2: +# - primary +# layer3: +# defaults: +# peer_port: 179 +# hold_time: 120s +# communities: +# vpn-only: "1234:1" +# NO_ADVERTISE: "65535:65282" +# metallb_peers: +# peer1: +# peer_address: 10.6.0.1 +# peer_asn: 64512 +# my_asn: 4200000000 +# communities: +# - vpn-only +# address_pool: +# - pool1 +# peer2: +# peer_address: 10.10.0.1 +# peer_asn: 64513 +# my_asn: 4200000000 +# communities: +# - NO_ADVERTISE +# address_pool: +# - pool2 + +argocd_enabled: false +# argocd_version: v2.8.0 +# argocd_namespace: argocd +# Default password: +# - https://argo-cd.readthedocs.io/en/stable/getting_started/#4-login-using-the-cli +# --- +# The initial password is autogenerated and stored in `argocd-initial-admin-secret` in the argocd namespace defined above. +# Using the argocd CLI the generated password can be automatically be fetched from the current kubectl context with the command: +# argocd admin initial-password -n argocd +# --- +# Use the following var to set admin password +# argocd_admin_password: "password" + +# The plugin manager for kubectl +krew_enabled: false +krew_root_dir: "/usr/local/krew" diff --git a/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-cluster.yml b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-cluster.yml new file mode 100644 index 0000000..69574c8 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-cluster.yml @@ -0,0 +1,382 @@ +--- +# Kubernetes configuration dirs and system namespace. +# Those are where all the additional config stuff goes +# the kubernetes normally puts in /srv/kubernetes. +# This puts them in a sane location and namespace. +# Editing those values will almost surely break something. +kube_config_dir: /etc/kubernetes +kube_script_dir: "{{ bin_dir }}/kubernetes-scripts" +kube_manifest_dir: "{{ kube_config_dir }}/manifests" + +# This is where all the cert scripts and certs will be located +kube_cert_dir: "{{ kube_config_dir }}/ssl" + +# This is where all of the bearer tokens will be stored +kube_token_dir: "{{ kube_config_dir }}/tokens" + +kube_api_anonymous_auth: true + +## Change this to use another Kubernetes version, e.g. a current beta release +kube_version: v1.28.2 + +# Where the binaries will be downloaded. +# Note: ensure that you've enough disk space (about 1G) +local_release_dir: "/tmp/releases" +# Random shifts for retrying failed ops like pushing/downloading +retry_stagger: 5 + +# This is the user that owns tha cluster installation. +kube_owner: kube + +# This is the group that the cert creation scripts chgrp the +# cert files to. Not really changeable... +kube_cert_group: kube-cert + +# Cluster Loglevel configuration +kube_log_level: 2 + +# Directory where credentials will be stored +credentials_dir: "{{ inventory_dir }}/credentials" + +## It is possible to activate / deactivate selected authentication methods (oidc, static token auth) +# kube_oidc_auth: false +# kube_token_auth: false + + +## Variables for OpenID Connect Configuration https://kubernetes.io/docs/admin/authentication/ +## To use OpenID you have to deploy additional an OpenID Provider (e.g Dex, Keycloak, ...) + +# kube_oidc_url: https:// ... +# kube_oidc_client_id: kubernetes +## Optional settings for OIDC +# kube_oidc_ca_file: "{{ kube_cert_dir }}/ca.pem" +# kube_oidc_username_claim: sub +# kube_oidc_username_prefix: 'oidc:' +# kube_oidc_groups_claim: groups +# kube_oidc_groups_prefix: 'oidc:' + +## Variables to control webhook authn/authz +# kube_webhook_token_auth: false +# kube_webhook_token_auth_url: https://... +# kube_webhook_token_auth_url_skip_tls_verify: false + +## For webhook authorization, authorization_modes must include Webhook +# kube_webhook_authorization: false +# kube_webhook_authorization_url: https://... +# kube_webhook_authorization_url_skip_tls_verify: false + +# Choose network plugin (cilium, calico, kube-ovn, weave or flannel. Use cni for generic cni plugin) +# Can also be set to 'cloud', which lets the cloud provider setup appropriate routing +kube_network_plugin: calico + +# Setting multi_networking to true will install Multus: https://github.com/k8snetworkplumbingwg/multus-cni +kube_network_plugin_multus: false + +# Kubernetes internal network for services, unused block of space. +kube_service_addresses: 10.233.0.0/18 + +# internal network. When used, it will assign IP +# addresses from this range to individual pods. +# This network must be unused in your network infrastructure! +kube_pods_subnet: 10.233.64.0/18 + +# internal network node size allocation (optional). This is the size allocated +# to each node for pod IP address allocation. Note that the number of pods per node is +# also limited by the kubelet_max_pods variable which defaults to 110. +# +# Example: +# Up to 64 nodes and up to 254 or kubelet_max_pods (the lowest of the two) pods per node: +# - kube_pods_subnet: 10.233.64.0/18 +# - kube_network_node_prefix: 24 +# - kubelet_max_pods: 110 +# +# Example: +# Up to 128 nodes and up to 126 or kubelet_max_pods (the lowest of the two) pods per node: +# - kube_pods_subnet: 10.233.64.0/18 +# - kube_network_node_prefix: 25 +# - kubelet_max_pods: 110 +kube_network_node_prefix: 24 + +# Configure Dual Stack networking (i.e. both IPv4 and IPv6) +enable_dual_stack_networks: false + +# Kubernetes internal network for IPv6 services, unused block of space. +# This is only used if enable_dual_stack_networks is set to true +# This provides 4096 IPv6 IPs +kube_service_addresses_ipv6: fd85:ee78:d8a6:8607::1000/116 + +# Internal network. When used, it will assign IPv6 addresses from this range to individual pods. +# This network must not already be in your network infrastructure! +# This is only used if enable_dual_stack_networks is set to true. +# This provides room for 256 nodes with 254 pods per node. +kube_pods_subnet_ipv6: fd85:ee78:d8a6:8607::1:0000/112 + +# IPv6 subnet size allocated to each for pods. +# This is only used if enable_dual_stack_networks is set to true +# This provides room for 254 pods per node. +kube_network_node_prefix_ipv6: 120 + +# The port the API Server will be listening on. +kube_apiserver_ip: "{{ kube_service_addresses | ipaddr('net') | ipaddr(1) | ipaddr('address') }}" +kube_apiserver_port: 6443 # (https) + +# Kube-proxy proxyMode configuration. +# Can be ipvs, iptables +kube_proxy_mode: ipvs + +# configure arp_ignore and arp_announce to avoid answering ARP queries from kube-ipvs0 interface +# must be set to true for MetalLB, kube-vip(ARP enabled) to work +kube_proxy_strict_arp: false + +# A string slice of values which specify the addresses to use for NodePorts. +# Values may be valid IP blocks (e.g. 1.2.3.0/24, 1.2.3.4/32). +# The default empty string slice ([]) means to use all local addresses. +# kube_proxy_nodeport_addresses_cidr is retained for legacy config +kube_proxy_nodeport_addresses: >- + {%- if kube_proxy_nodeport_addresses_cidr is defined -%} + [{{ kube_proxy_nodeport_addresses_cidr }}] + {%- else -%} + [] + {%- endif -%} + +# If non-empty, will use this string as identification instead of the actual hostname +# kube_override_hostname: >- +# {%- if cloud_provider is defined and cloud_provider in ['aws'] -%} +# {%- else -%} +# {{ inventory_hostname }} +# {%- endif -%} + +## Encrypting Secret Data at Rest +kube_encrypt_secret_data: false + +# Graceful Node Shutdown (Kubernetes >= 1.21.0), see https://kubernetes.io/blog/2021/04/21/graceful-node-shutdown-beta/ +# kubelet_shutdown_grace_period had to be greater than kubelet_shutdown_grace_period_critical_pods to allow +# non-critical podsa to also terminate gracefully +# kubelet_shutdown_grace_period: 60s +# kubelet_shutdown_grace_period_critical_pods: 20s + +# DNS configuration. +# Kubernetes cluster name, also will be used as DNS domain +cluster_name: cluster.local +# Subdomains of DNS domain to be resolved via /etc/resolv.conf for hostnet pods +ndots: 2 +# dns_timeout: 2 +# dns_attempts: 2 +# Custom search domains to be added in addition to the default cluster search domains +# searchdomains: +# - svc.{{ cluster_name }} +# - default.svc.{{ cluster_name }} +# Remove default cluster search domains (``default.svc.{{ dns_domain }}, svc.{{ dns_domain }}``). +# remove_default_searchdomains: false +# Can be coredns, coredns_dual, manual or none +dns_mode: coredns +# Set manual server if using a custom cluster DNS server +# manual_dns_server: 10.x.x.x +# Enable nodelocal dns cache +enable_nodelocaldns: true +enable_nodelocaldns_secondary: false +nodelocaldns_ip: 169.254.25.10 +nodelocaldns_health_port: 9254 +nodelocaldns_second_health_port: 9256 +nodelocaldns_bind_metrics_host_ip: false +nodelocaldns_secondary_skew_seconds: 5 +# nodelocaldns_external_zones: +# - zones: +# - example.com +# - example.io:1053 +# nameservers: +# - 1.1.1.1 +# - 2.2.2.2 +# cache: 5 +# - zones: +# - https://mycompany.local:4453 +# nameservers: +# - 192.168.0.53 +# cache: 0 +# - zones: +# - mydomain.tld +# nameservers: +# - 10.233.0.3 +# cache: 5 +# rewrite: +# - name website.tld website.namespace.svc.cluster.local +# Enable k8s_external plugin for CoreDNS +enable_coredns_k8s_external: false +coredns_k8s_external_zone: k8s_external.local +# Enable endpoint_pod_names option for kubernetes plugin +enable_coredns_k8s_endpoint_pod_names: false +# Set forward options for upstream DNS servers in coredns (and nodelocaldns) config +# dns_upstream_forward_extra_opts: +# policy: sequential +# Apply extra options to coredns kubernetes plugin +# coredns_kubernetes_extra_opts: +# - 'fallthrough example.local' +# Forward extra domains to the coredns kubernetes plugin +# coredns_kubernetes_extra_domains: '' + +# Can be docker_dns, host_resolvconf or none +resolvconf_mode: host_resolvconf +# Deploy netchecker app to verify DNS resolve as an HTTP service +deploy_netchecker: false +# Ip address of the kubernetes skydns service +skydns_server: "{{ kube_service_addresses | ipaddr('net') | ipaddr(3) | ipaddr('address') }}" +skydns_server_secondary: "{{ kube_service_addresses | ipaddr('net') | ipaddr(4) | ipaddr('address') }}" +dns_domain: "{{ cluster_name }}" + +## Container runtime +## docker for docker, crio for cri-o and containerd for containerd. +## Default: containerd +container_manager: containerd + +# Additional container runtimes +kata_containers_enabled: false + +kubeadm_certificate_key: "{{ lookup('password', credentials_dir + '/kubeadm_certificate_key.creds length=64 chars=hexdigits') | lower }}" + +# K8s image pull policy (imagePullPolicy) +k8s_image_pull_policy: IfNotPresent + +# audit log for kubernetes +kubernetes_audit: false + +# define kubelet config dir for dynamic kubelet +# kubelet_config_dir: +default_kubelet_config_dir: "{{ kube_config_dir }}/dynamic_kubelet_dir" + +# pod security policy (RBAC must be enabled either by having 'RBAC' in authorization_modes or kubeadm enabled) +podsecuritypolicy_enabled: false + +# Custom PodSecurityPolicySpec for restricted policy +# podsecuritypolicy_restricted_spec: {} + +# Custom PodSecurityPolicySpec for privileged policy +# podsecuritypolicy_privileged_spec: {} + +# Make a copy of kubeconfig on the host that runs Ansible in {{ inventory_dir }}/artifacts +# kubeconfig_localhost: false +# Use ansible_host as external api ip when copying over kubeconfig. +# kubeconfig_localhost_ansible_host: false +# Download kubectl onto the host that runs Ansible in {{ bin_dir }} +# kubectl_localhost: false + +# A comma separated list of levels of node allocatable enforcement to be enforced by kubelet. +# Acceptable options are 'pods', 'system-reserved', 'kube-reserved' and ''. Default is "". +# kubelet_enforce_node_allocatable: pods + +## Set runtime and kubelet cgroups when using systemd as cgroup driver (default) +# kubelet_runtime_cgroups: "/{{ kube_service_cgroups }}/{{ container_manager }}.service" +# kubelet_kubelet_cgroups: "/{{ kube_service_cgroups }}/kubelet.service" + +## Set runtime and kubelet cgroups when using cgroupfs as cgroup driver +# kubelet_runtime_cgroups_cgroupfs: "/system.slice/{{ container_manager }}.service" +# kubelet_kubelet_cgroups_cgroupfs: "/system.slice/kubelet.service" + +# Optionally reserve this space for kube daemons. +# kube_reserved: false +## Uncomment to override default values +## The following two items need to be set when kube_reserved is true +# kube_reserved_cgroups_for_service_slice: kube.slice +# kube_reserved_cgroups: "/{{ kube_reserved_cgroups_for_service_slice }}" +# kube_memory_reserved: 256Mi +# kube_cpu_reserved: 100m +# kube_ephemeral_storage_reserved: 2Gi +# kube_pid_reserved: "1000" +# Reservation for master hosts +# kube_master_memory_reserved: 512Mi +# kube_master_cpu_reserved: 200m +# kube_master_ephemeral_storage_reserved: 2Gi +# kube_master_pid_reserved: "1000" + +## Optionally reserve resources for OS system daemons. +# system_reserved: true +## Uncomment to override default values +## The following two items need to be set when system_reserved is true +# system_reserved_cgroups_for_service_slice: system.slice +# system_reserved_cgroups: "/{{ system_reserved_cgroups_for_service_slice }}" +# system_memory_reserved: 512Mi +# system_cpu_reserved: 500m +# system_ephemeral_storage_reserved: 2Gi +## Reservation for master hosts +# system_master_memory_reserved: 256Mi +# system_master_cpu_reserved: 250m +# system_master_ephemeral_storage_reserved: 2Gi + +## Eviction Thresholds to avoid system OOMs +# https://kubernetes.io/docs/tasks/administer-cluster/reserve-compute-resources/#eviction-thresholds +# eviction_hard: {} +# eviction_hard_control_plane: {} + +# An alternative flexvolume plugin directory +# kubelet_flexvolumes_plugins_dir: /usr/libexec/kubernetes/kubelet-plugins/volume/exec + +## Supplementary addresses that can be added in kubernetes ssl keys. +## That can be useful for example to setup a keepalived virtual IP +# supplementary_addresses_in_ssl_keys: [10.0.0.1, 10.0.0.2, 10.0.0.3] + +## Running on top of openstack vms with cinder enabled may lead to unschedulable pods due to NoVolumeZoneConflict restriction in kube-scheduler. +## See https://github.com/kubernetes-sigs/kubespray/issues/2141 +## Set this variable to true to get rid of this issue +volume_cross_zone_attachment: false +## Add Persistent Volumes Storage Class for corresponding cloud provider (supported: in-tree OpenStack, Cinder CSI, +## AWS EBS CSI, Azure Disk CSI, GCP Persistent Disk CSI) +persistent_volumes_enabled: false + +## Container Engine Acceleration +## Enable container acceleration feature, for example use gpu acceleration in containers +# nvidia_accelerator_enabled: true +## Nvidia GPU driver install. Install will by done by a (init) pod running as a daemonset. +## Important: if you use Ubuntu then you should set in all.yml 'docker_storage_options: -s overlay2' +## Array with nvida_gpu_nodes, leave empty or comment if you don't want to install drivers. +## Labels and taints won't be set to nodes if they are not in the array. +# nvidia_gpu_nodes: +# - kube-gpu-001 +# nvidia_driver_version: "384.111" +## flavor can be tesla or gtx +# nvidia_gpu_flavor: gtx +## NVIDIA driver installer images. Change them if you have trouble accessing gcr.io. +# nvidia_driver_install_centos_container: atzedevries/nvidia-centos-driver-installer:2 +# nvidia_driver_install_ubuntu_container: gcr.io/google-containers/ubuntu-nvidia-driver-installer@sha256:7df76a0f0a17294e86f691c81de6bbb7c04a1b4b3d4ea4e7e2cccdc42e1f6d63 +## NVIDIA GPU device plugin image. +# nvidia_gpu_device_plugin_container: "registry.k8s.io/nvidia-gpu-device-plugin@sha256:0842734032018be107fa2490c98156992911e3e1f2a21e059ff0105b07dd8e9e" + +## Support tls min version, Possible values: VersionTLS10, VersionTLS11, VersionTLS12, VersionTLS13. +# tls_min_version: "" + +## Support tls cipher suites. +# tls_cipher_suites: {} +# - TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA +# - TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256 +# - TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 +# - TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA +# - TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 +# - TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305 +# - TLS_ECDHE_ECDSA_WITH_RC4_128_SHA +# - TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256 +# - TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 +# - TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 +# - TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305 +# - TLS_ECDHE_RSA_WITH_RC4_128_SHA +# - TLS_RSA_WITH_3DES_EDE_CBC_SHA +# - TLS_RSA_WITH_AES_128_CBC_SHA +# - TLS_RSA_WITH_AES_128_CBC_SHA256 +# - TLS_RSA_WITH_AES_128_GCM_SHA256 +# - TLS_RSA_WITH_AES_256_CBC_SHA +# - TLS_RSA_WITH_AES_256_GCM_SHA384 +# - TLS_RSA_WITH_RC4_128_SHA + +## Amount of time to retain events. (default 1h0m0s) +event_ttl_duration: "1h0m0s" + +## Automatically renew K8S control plane certificates on first Monday of each month +auto_renew_certificates: false +# First Monday of each month +# auto_renew_certificates_systemd_calendar: "Mon *-*-1,2,3,4,5,6,7 03:{{ groups['kube_control_plane'].index(inventory_hostname) }}0:00" + +# kubeadm patches path +kubeadm_patches: + enabled: false + source_dir: "{{ inventory_dir }}/patches" + dest_dir: "{{ kube_config_dir }}/patches" diff --git a/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-calico.yml b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-calico.yml new file mode 100644 index 0000000..cc0499d --- /dev/null +++ b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-calico.yml @@ -0,0 +1,131 @@ +--- +# see roles/network_plugin/calico/defaults/main.yml + +# the default value of name +calico_cni_name: k8s-pod-network + +## With calico it is possible to distributed routes with border routers of the datacenter. +## Warning : enabling router peering will disable calico's default behavior ('node mesh'). +## The subnets of each nodes will be distributed by the datacenter router +# peer_with_router: false + +# Enables Internet connectivity from containers +# nat_outgoing: true + +# Enables Calico CNI "host-local" IPAM plugin +# calico_ipam_host_local: true + +# add default ippool name +# calico_pool_name: "default-pool" + +# add default ippool blockSize (defaults kube_network_node_prefix) +calico_pool_blocksize: 26 + +# add default ippool CIDR (must be inside kube_pods_subnet, defaults to kube_pods_subnet otherwise) +# calico_pool_cidr: 1.2.3.4/5 + +# add default ippool CIDR to CNI config +# calico_cni_pool: true + +# Add default IPV6 IPPool CIDR. Must be inside kube_pods_subnet_ipv6. Defaults to kube_pods_subnet_ipv6 if not set. +# calico_pool_cidr_ipv6: fd85:ee78:d8a6:8607::1:0000/112 + +# Add default IPV6 IPPool CIDR to CNI config +# calico_cni_pool_ipv6: true + +# Global as_num (/calico/bgp/v1/global/as_num) +# global_as_num: "64512" + +# If doing peering with node-assigned asn where the globas does not match your nodes, you want this +# to be true. All other cases, false. +# calico_no_global_as_num: false + +# You can set MTU value here. If left undefined or empty, it will +# not be specified in calico CNI config, so Calico will use built-in +# defaults. The value should be a number, not a string. +# calico_mtu: 1500 + +# Configure the MTU to use for workload interfaces and tunnels. +# - If Wireguard is enabled, subtract 60 from your network MTU (i.e 1500-60=1440) +# - Otherwise, if VXLAN or BPF mode is enabled, subtract 50 from your network MTU (i.e. 1500-50=1450) +# - Otherwise, if IPIP is enabled, subtract 20 from your network MTU (i.e. 1500-20=1480) +# - Otherwise, if not using any encapsulation, set to your network MTU (i.e. 1500) +# calico_veth_mtu: 1440 + +# Advertise Cluster IPs +# calico_advertise_cluster_ips: true + +# Advertise Service External IPs +# calico_advertise_service_external_ips: +# - x.x.x.x/24 +# - y.y.y.y/32 + +# Advertise Service LoadBalancer IPs +# calico_advertise_service_loadbalancer_ips: +# - x.x.x.x/24 +# - y.y.y.y/16 + +# Choose data store type for calico: "etcd" or "kdd" (kubernetes datastore) +# calico_datastore: "kdd" + +# Choose Calico iptables backend: "Legacy", "Auto" or "NFT" +# calico_iptables_backend: "Auto" + +# Use typha (only with kdd) +# typha_enabled: false + +# Generate TLS certs for secure typha<->calico-node communication +# typha_secure: false + +# Scaling typha: 1 replica per 100 nodes is adequate +# Number of typha replicas +# typha_replicas: 1 + +# Set max typha connections +# typha_max_connections_lower_limit: 300 + +# Set calico network backend: "bird", "vxlan" or "none" +# bird enable BGP routing, required for ipip and no encapsulation modes +# calico_network_backend: vxlan + +# IP in IP and VXLAN is mutualy exclusive modes. +# set IP in IP encapsulation mode: "Always", "CrossSubnet", "Never" +# calico_ipip_mode: 'Never' + +# set VXLAN encapsulation mode: "Always", "CrossSubnet", "Never" +# calico_vxlan_mode: 'Always' + +# set VXLAN port and VNI +# calico_vxlan_vni: 4096 +# calico_vxlan_port: 4789 + +# Enable eBPF mode +# calico_bpf_enabled: false + +# If you want to use non default IP_AUTODETECTION_METHOD, IP6_AUTODETECTION_METHOD for calico node set this option to one of: +# * can-reach=DESTINATION +# * interface=INTERFACE-REGEX +# see https://docs.projectcalico.org/reference/node/configuration +# calico_ip_auto_method: "interface=eth.*" +# calico_ip6_auto_method: "interface=eth.*" + +# Set FELIX_MTUIFACEPATTERN, Pattern used to discover the host’s interface for MTU auto-detection. +# see https://projectcalico.docs.tigera.io/reference/felix/configuration +# calico_felix_mtu_iface_pattern: "^((en|wl|ww|sl|ib)[opsx].*|(eth|wlan|wwan).*)" + +# Choose the iptables insert mode for Calico: "Insert" or "Append". +# calico_felix_chaininsertmode: Insert + +# If you want use the default route interface when you use multiple interface with dynamique route (iproute2) +# see https://docs.projectcalico.org/reference/node/configuration : FELIX_DEVICEROUTESOURCEADDRESS +# calico_use_default_route_src_ipaddr: false + +# Enable calico traffic encryption with wireguard +# calico_wireguard_enabled: false + +# Under certain situations liveness and readiness probes may need tunning +# calico_node_livenessprobe_timeout: 10 +# calico_node_readinessprobe_timeout: 10 + +# Calico apiserver (only with kdd) +# calico_apiserver_enabled: false diff --git a/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-cilium.yml b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-cilium.yml new file mode 100644 index 0000000..a170484 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-cilium.yml @@ -0,0 +1,264 @@ +--- +# cilium_version: "v1.12.1" + +# Log-level +# cilium_debug: false + +# cilium_mtu: "" +# cilium_enable_ipv4: true +# cilium_enable_ipv6: false + +# Cilium agent health port +# cilium_agent_health_port: "9879" + +# Identity allocation mode selects how identities are shared between cilium +# nodes by setting how they are stored. The options are "crd" or "kvstore". +# - "crd" stores identities in kubernetes as CRDs (custom resource definition). +# These can be queried with: +# `kubectl get ciliumid` +# - "kvstore" stores identities in an etcd kvstore. +# - In order to support External Workloads, "crd" is required +# - Ref: https://docs.cilium.io/en/stable/gettingstarted/external-workloads/#setting-up-support-for-external-workloads-beta +# - KVStore operations are only required when cilium-operator is running with any of the below options: +# - --synchronize-k8s-services +# - --synchronize-k8s-nodes +# - --identity-allocation-mode=kvstore +# - Ref: https://docs.cilium.io/en/stable/internals/cilium_operator/#kvstore-operations +# cilium_identity_allocation_mode: kvstore + +# Etcd SSL dirs +# cilium_cert_dir: /etc/cilium/certs +# kube_etcd_cacert_file: ca.pem +# kube_etcd_cert_file: cert.pem +# kube_etcd_key_file: cert-key.pem + +# Limits for apps +# cilium_memory_limit: 500M +# cilium_cpu_limit: 500m +# cilium_memory_requests: 64M +# cilium_cpu_requests: 100m + +# Overlay Network Mode +# cilium_tunnel_mode: vxlan +# Optional features +# cilium_enable_prometheus: false +# Enable if you want to make use of hostPort mappings +# cilium_enable_portmap: false +# Monitor aggregation level (none/low/medium/maximum) +# cilium_monitor_aggregation: medium +# The monitor aggregation flags determine which TCP flags which, upon the +# first observation, cause monitor notifications to be generated. +# +# Only effective when monitor aggregation is set to "medium" or higher. +# cilium_monitor_aggregation_flags: "all" +# Kube Proxy Replacement mode (strict/partial) +# cilium_kube_proxy_replacement: partial + +# If upgrading from Cilium < 1.5, you may want to override some of these options +# to prevent service disruptions. See also: +# http://docs.cilium.io/en/stable/install/upgrade/#changes-that-may-require-action +# cilium_preallocate_bpf_maps: false + +# `cilium_tofqdns_enable_poller` is deprecated in 1.8, removed in 1.9 +# cilium_tofqdns_enable_poller: false + +# `cilium_enable_legacy_services` is deprecated in 1.6, removed in 1.9 +# cilium_enable_legacy_services: false + +# Unique ID of the cluster. Must be unique across all conneted clusters and +# in the range of 1 and 255. Only relevant when building a mesh of clusters. +# This value is not defined by default +# cilium_cluster_id: + +# Deploy cilium even if kube_network_plugin is not cilium. +# This enables to deploy cilium alongside another CNI to replace kube-proxy. +# cilium_deploy_additionally: false + +# Auto direct nodes routes can be used to advertise pods routes in your cluster +# without any tunelling (with `cilium_tunnel_mode` sets to `disabled`). +# This works only if you have a L2 connectivity between all your nodes. +# You wil also have to specify the variable `cilium_native_routing_cidr` to +# make this work. Please refer to the cilium documentation for more +# information about this kind of setups. +# cilium_auto_direct_node_routes: false + +# Allows to explicitly specify the IPv4 CIDR for native routing. +# When specified, Cilium assumes networking for this CIDR is preconfigured and +# hands traffic destined for that range to the Linux network stack without +# applying any SNAT. +# Generally speaking, specifying a native routing CIDR implies that Cilium can +# depend on the underlying networking stack to route packets to their +# destination. To offer a concrete example, if Cilium is configured to use +# direct routing and the Kubernetes CIDR is included in the native routing CIDR, +# the user must configure the routes to reach pods, either manually or by +# setting the auto-direct-node-routes flag. +# cilium_native_routing_cidr: "" + +# Allows to explicitly specify the IPv6 CIDR for native routing. +# cilium_native_routing_cidr_ipv6: "" + +# Enable transparent network encryption. +# cilium_encryption_enabled: false + +# Encryption method. Can be either ipsec or wireguard. +# Only effective when `cilium_encryption_enabled` is set to true. +# cilium_encryption_type: "ipsec" + +# Enable encryption for pure node to node traffic. +# This option is only effective when `cilium_encryption_type` is set to `ipsec`. +# cilium_ipsec_node_encryption: false + +# If your kernel or distribution does not support WireGuard, Cilium agent can be configured to fall back on the user-space implementation. +# When this flag is enabled and Cilium detects that the kernel has no native support for WireGuard, +# it will fallback on the wireguard-go user-space implementation of WireGuard. +# This option is only effective when `cilium_encryption_type` is set to `wireguard`. +# cilium_wireguard_userspace_fallback: false + +# IP Masquerade Agent +# https://docs.cilium.io/en/stable/concepts/networking/masquerading/ +# By default, all packets from a pod destined to an IP address outside of the cilium_native_routing_cidr range are masqueraded +# cilium_ip_masq_agent_enable: false + +### A packet sent from a pod to a destination which belongs to any CIDR from the nonMasqueradeCIDRs is not going to be masqueraded +# cilium_non_masquerade_cidrs: +# - 10.0.0.0/8 +# - 172.16.0.0/12 +# - 192.168.0.0/16 +# - 100.64.0.0/10 +# - 192.0.0.0/24 +# - 192.0.2.0/24 +# - 192.88.99.0/24 +# - 198.18.0.0/15 +# - 198.51.100.0/24 +# - 203.0.113.0/24 +# - 240.0.0.0/4 +### Indicates whether to masquerade traffic to the link local prefix. +### If the masqLinkLocal is not set or set to false, then 169.254.0.0/16 is appended to the non-masquerade CIDRs list. +# cilium_masq_link_local: false +### A time interval at which the agent attempts to reload config from disk +# cilium_ip_masq_resync_interval: 60s + +# Hubble +### Enable Hubble without install +# cilium_enable_hubble: false +### Enable Hubble Metrics +# cilium_enable_hubble_metrics: false +### if cilium_enable_hubble_metrics: true +# cilium_hubble_metrics: {} +# - dns +# - drop +# - tcp +# - flow +# - icmp +# - http +### Enable Hubble install +# cilium_hubble_install: false +### Enable auto generate certs if cilium_hubble_install: true +# cilium_hubble_tls_generate: false + +# IP address management mode for v1.9+. +# https://docs.cilium.io/en/v1.9/concepts/networking/ipam/ +# cilium_ipam_mode: kubernetes + +# Extra arguments for the Cilium agent +# cilium_agent_custom_args: [] + +# For adding and mounting extra volumes to the cilium agent +# cilium_agent_extra_volumes: [] +# cilium_agent_extra_volume_mounts: [] + +# cilium_agent_extra_env_vars: [] + +# cilium_operator_replicas: 2 + +# The address at which the cillium operator bind health check api +# cilium_operator_api_serve_addr: "127.0.0.1:9234" + +## A dictionary of extra config variables to add to cilium-config, formatted like: +## cilium_config_extra_vars: +## var1: "value1" +## var2: "value2" +# cilium_config_extra_vars: {} + +# For adding and mounting extra volumes to the cilium operator +# cilium_operator_extra_volumes: [] +# cilium_operator_extra_volume_mounts: [] + +# Extra arguments for the Cilium Operator +# cilium_operator_custom_args: [] + +# Name of the cluster. Only relevant when building a mesh of clusters. +# cilium_cluster_name: default + +# Make Cilium take ownership over the `/etc/cni/net.d` directory on the node, renaming all non-Cilium CNI configurations to `*.cilium_bak`. +# This ensures no Pods can be scheduled using other CNI plugins during Cilium agent downtime. +# Available for Cilium v1.10 and up. +# cilium_cni_exclusive: true + +# Configure the log file for CNI logging with retention policy of 7 days. +# Disable CNI file logging by setting this field to empty explicitly. +# Available for Cilium v1.12 and up. +# cilium_cni_log_file: "/var/run/cilium/cilium-cni.log" + +# -- Configure cgroup related configuration +# -- Enable auto mount of cgroup2 filesystem. +# When `cilium_cgroup_auto_mount` is enabled, cgroup2 filesystem is mounted at +# `cilium_cgroup_host_root` path on the underlying host and inside the cilium agent pod. +# If users disable `cilium_cgroup_auto_mount`, it's expected that users have mounted +# cgroup2 filesystem at the specified `cilium_cgroup_auto_mount` volume, and then the +# volume will be mounted inside the cilium agent pod at the same path. +# Available for Cilium v1.11 and up +# cilium_cgroup_auto_mount: true +# -- Configure cgroup root where cgroup2 filesystem is mounted on the host +# cilium_cgroup_host_root: "/run/cilium/cgroupv2" + +# Specifies the ratio (0.0-1.0) of total system memory to use for dynamic +# sizing of the TCP CT, non-TCP CT, NAT and policy BPF maps. +# cilium_bpf_map_dynamic_size_ratio: "0.0" + +# -- Enables masquerading of IPv4 traffic leaving the node from endpoints. +# Available for Cilium v1.10 and up +# cilium_enable_ipv4_masquerade: true +# -- Enables masquerading of IPv6 traffic leaving the node from endpoints. +# Available for Cilium v1.10 and up +# cilium_enable_ipv6_masquerade: true + +# -- Enable native IP masquerade support in eBPF +# cilium_enable_bpf_masquerade: false + +# -- Configure whether direct routing mode should route traffic via +# host stack (true) or directly and more efficiently out of BPF (false) if +# the kernel supports it. The latter has the implication that it will also +# bypass netfilter in the host namespace. +# cilium_enable_host_legacy_routing: true + +# -- Enable use of the remote node identity. +# ref: https://docs.cilium.io/en/v1.7/install/upgrade/#configmap-remote-node-identity +# cilium_enable_remote_node_identity: true + +# -- Enable the use of well-known identities. +# cilium_enable_well_known_identities: false + +# cilium_enable_bpf_clock_probe: true + +# -- Whether to enable CNP status updates. +# cilium_disable_cnp_status_updates: true + +# A list of extra rules variables to add to clusterrole for cilium operator, formatted like: +# cilium_clusterrole_rules_operator_extra_vars: +# - apiGroups: +# - '""' +# resources: +# - pods +# verbs: +# - delete +# - apiGroups: +# - '""' +# resources: +# - nodes +# verbs: +# - list +# - watch +# resourceNames: +# - toto +# cilium_clusterrole_rules_operator_extra_vars: [] diff --git a/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-flannel.yml b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-flannel.yml new file mode 100644 index 0000000..64d20a8 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-flannel.yml @@ -0,0 +1,18 @@ +# see roles/network_plugin/flannel/defaults/main.yml + +## interface that should be used for flannel operations +## This is actually an inventory cluster-level item +# flannel_interface: + +## Select interface that should be used for flannel operations by regexp on Name or IP +## This is actually an inventory cluster-level item +## example: select interface with ip from net 10.0.0.0/23 +## single quote and escape backslashes +# flannel_interface_regexp: '10\\.0\\.[0-2]\\.\\d{1,3}' + +# You can choose what type of flannel backend to use: 'vxlan', 'host-gw' or 'wireguard' +# please refer to flannel's docs : https://github.com/coreos/flannel/blob/master/README.md +# flannel_backend_type: "vxlan" +# flannel_vxlan_vni: 1 +# flannel_vxlan_port: 8472 +# flannel_vxlan_direct_routing: false diff --git a/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-kube-ovn.yml b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-kube-ovn.yml new file mode 100644 index 0000000..c241a76 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-kube-ovn.yml @@ -0,0 +1,63 @@ +--- + +# geneve or vlan +kube_ovn_network_type: geneve + +# geneve, vxlan or stt. ATTENTION: some networkpolicy cannot take effect when using vxlan and stt need custom compile ovs kernel module +kube_ovn_tunnel_type: geneve + +## The nic to support container network can be a nic name or a group of regex separated by comma e.g: 'enp6s0f0,eth.*', if empty will use the nic that the default route use. +# kube_ovn_iface: eth1 +## The MTU used by pod iface in overlay networks (default iface MTU - 100) +# kube_ovn_mtu: 1333 + +## Enable hw-offload, disable traffic mirror and set the iface to the physical port. Make sure that there is an IP address bind to the physical port. +kube_ovn_hw_offload: false +# traffic mirror +kube_ovn_traffic_mirror: false + +# kube_ovn_pool_cidr_ipv6: fd85:ee78:d8a6:8607::1:0000/112 +# kube_ovn_default_interface_name: eth0 + +kube_ovn_external_address: 8.8.8.8 +kube_ovn_external_address_ipv6: 2400:3200::1 +kube_ovn_external_dns: alauda.cn + +# kube_ovn_default_gateway: 10.233.64.1,fd85:ee78:d8a6:8607::1:0 +kube_ovn_default_gateway_check: true +kube_ovn_default_logical_gateway: false +# kube_ovn_default_exclude_ips: 10.16.0.1 +kube_ovn_node_switch_cidr: 100.64.0.0/16 +kube_ovn_node_switch_cidr_ipv6: fd00:100:64::/64 + +## vlan config, set default interface name and vlan id +# kube_ovn_default_interface_name: eth0 +kube_ovn_default_vlan_id: 100 +kube_ovn_vlan_name: product + +## pod nic type, support: veth-pair or internal-port +kube_ovn_pod_nic_type: veth_pair + +## Enable load balancer +kube_ovn_enable_lb: true + +## Enable network policy support +kube_ovn_enable_np: true + +## Enable external vpc support +kube_ovn_enable_external_vpc: true + +## Enable checksum +kube_ovn_encap_checksum: true + +## enable ssl +kube_ovn_enable_ssl: false + +## dpdk +kube_ovn_dpdk_enabled: false + +## enable interconnection to an existing IC database server. +kube_ovn_ic_enable: false +kube_ovn_ic_autoroute: true +kube_ovn_ic_dbhost: "127.0.0.1" +kube_ovn_ic_zone: "kubernetes" diff --git a/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-kube-router.yml b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-kube-router.yml new file mode 100644 index 0000000..e4dfcc9 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-kube-router.yml @@ -0,0 +1,64 @@ +# See roles/network_plugin/kube-router//defaults/main.yml + +# Enables Pod Networking -- Advertises and learns the routes to Pods via iBGP +# kube_router_run_router: true + +# Enables Network Policy -- sets up iptables to provide ingress firewall for pods +# kube_router_run_firewall: true + +# Enables Service Proxy -- sets up IPVS for Kubernetes Services +# see docs/kube-router.md "Caveats" section +# kube_router_run_service_proxy: false + +# Add Cluster IP of the service to the RIB so that it gets advertises to the BGP peers. +# kube_router_advertise_cluster_ip: false + +# Add External IP of service to the RIB so that it gets advertised to the BGP peers. +# kube_router_advertise_external_ip: false + +# Add LoadBalancer IP of service status as set by the LB provider to the RIB so that it gets advertised to the BGP peers. +# kube_router_advertise_loadbalancer_ip: false + +# Adjust manifest of kube-router daemonset template with DSR needed changes +# kube_router_enable_dsr: false + +# Array of arbitrary extra arguments to kube-router, see +# https://github.com/cloudnativelabs/kube-router/blob/master/docs/user-guide.md +# kube_router_extra_args: [] + +# ASN number of the cluster, used when communicating with external BGP routers +# kube_router_cluster_asn: ~ + +# ASN numbers of the BGP peer to which cluster nodes will advertise cluster ip and node's pod cidr. +# kube_router_peer_router_asns: ~ + +# The ip address of the external router to which all nodes will peer and advertise the cluster ip and pod cidr's. +# kube_router_peer_router_ips: ~ + +# The remote port of the external BGP to which all nodes will peer. If not set, default BGP port (179) will be used. +# kube_router_peer_router_ports: ~ + +# Setups node CNI to allow hairpin mode, requires node reboots, see +# https://github.com/cloudnativelabs/kube-router/blob/master/docs/user-guide.md#hairpin-mode +# kube_router_support_hairpin_mode: false + +# Select DNS Policy ClusterFirstWithHostNet, ClusterFirst, etc. +# kube_router_dns_policy: ClusterFirstWithHostNet + +# Array of annotations for master +# kube_router_annotations_master: [] + +# Array of annotations for every node +# kube_router_annotations_node: [] + +# Array of common annotations for every node +# kube_router_annotations_all: [] + +# Enables scraping kube-router metrics with Prometheus +# kube_router_enable_metrics: false + +# Path to serve Prometheus metrics on +# kube_router_metrics_path: /metrics + +# Prometheus metrics port to use +# kube_router_metrics_port: 9255 diff --git a/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-macvlan.yml b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-macvlan.yml new file mode 100644 index 0000000..d2534e7 --- /dev/null +++ b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-macvlan.yml @@ -0,0 +1,6 @@ +--- +# private interface, on a l2-network +macvlan_interface: "eth1" + +# Enable nat in default gateway network interface +enable_nat_default_gateway: true diff --git a/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-weave.yml b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-weave.yml new file mode 100644 index 0000000..269a77c --- /dev/null +++ b/kubespray/inventory/sample/group_vars/k8s_cluster/k8s-net-weave.yml @@ -0,0 +1,64 @@ +# see roles/network_plugin/weave/defaults/main.yml + +# Weave's network password for encryption, if null then no network encryption. +# weave_password: ~ + +# If set to 1, disable checking for new Weave Net versions (default is blank, +# i.e. check is enabled) +# weave_checkpoint_disable: false + +# Soft limit on the number of connections between peers. Defaults to 100. +# weave_conn_limit: 100 + +# Weave Net defaults to enabling hairpin on the bridge side of the veth pair +# for containers attached. If you need to disable hairpin, e.g. your kernel is +# one of those that can panic if hairpin is enabled, then you can disable it by +# setting `HAIRPIN_MODE=false`. +# weave_hairpin_mode: true + +# The range of IP addresses used by Weave Net and the subnet they are placed in +# (CIDR format; default 10.32.0.0/12) +# weave_ipalloc_range: "{{ kube_pods_subnet }}" + +# Set to 0 to disable Network Policy Controller (default is on) +# weave_expect_npc: "{{ enable_network_policy }}" + +# List of addresses of peers in the Kubernetes cluster (default is to fetch the +# list from the api-server) +# weave_kube_peers: ~ + +# Set the initialization mode of the IP Address Manager (defaults to consensus +# amongst the KUBE_PEERS) +# weave_ipalloc_init: ~ + +# Set the IP address used as a gateway from the Weave network to the host +# network - this is useful if you are configuring the addon as a static pod. +# weave_expose_ip: ~ + +# Address and port that the Weave Net daemon will serve Prometheus-style +# metrics on (defaults to 0.0.0.0:6782) +# weave_metrics_addr: ~ + +# Address and port that the Weave Net daemon will serve status requests on +# (defaults to disabled) +# weave_status_addr: ~ + +# Weave Net defaults to 1376 bytes, but you can set a smaller size if your +# underlying network has a tighter limit, or set a larger size for better +# performance if your network supports jumbo frames (e.g. 8916) +# weave_mtu: 1376 + +# Set to 1 to preserve the client source IP address when accessing Service +# annotated with `service.spec.externalTrafficPolicy=Local`. The feature works +# only with Weave IPAM (default). +# weave_no_masq_local: true + +# set to nft to use nftables backend for iptables (default is iptables) +# weave_iptables_backend: iptables + +# Extra variables that passing to launch.sh, useful for enabling seed mode, see +# https://www.weave.works/docs/net/latest/tasks/ipam/ipam/ +# weave_extra_args: ~ + +# Extra variables for weave_npc that passing to launch.sh, useful for change log level, ex --log-level=error +# weave_npc_extra_args: ~ diff --git a/kubespray/inventory/sample/inventory.ini b/kubespray/inventory/sample/inventory.ini new file mode 100644 index 0000000..99a6309 --- /dev/null +++ b/kubespray/inventory/sample/inventory.ini @@ -0,0 +1,38 @@ +# ## Configure 'ip' variable to bind kubernetes services on a +# ## different ip than the default iface +# ## We should set etcd_member_name for etcd cluster. The node that is not a etcd member do not need to set the value, or can set the empty string value. +[all] +# node1 ansible_host=95.54.0.12 # ip=10.3.0.1 etcd_member_name=etcd1 +# node2 ansible_host=95.54.0.13 # ip=10.3.0.2 etcd_member_name=etcd2 +# node3 ansible_host=95.54.0.14 # ip=10.3.0.3 etcd_member_name=etcd3 +# node4 ansible_host=95.54.0.15 # ip=10.3.0.4 etcd_member_name=etcd4 +# node5 ansible_host=95.54.0.16 # ip=10.3.0.5 etcd_member_name=etcd5 +# node6 ansible_host=95.54.0.17 # ip=10.3.0.6 etcd_member_name=etcd6 + +# ## configure a bastion host if your nodes are not directly reachable +# [bastion] +# bastion ansible_host=x.x.x.x ansible_user=some_user + +[kube_control_plane] +# node1 +# node2 +# node3 + +[etcd] +# node1 +# node2 +# node3 + +[kube_node] +# node2 +# node3 +# node4 +# node5 +# node6 + +[calico_rr] + +[k8s_cluster:children] +kube_control_plane +kube_node +calico_rr diff --git a/kubespray/inventory/sample/patches/kube-controller-manager+merge.yaml b/kubespray/inventory/sample/patches/kube-controller-manager+merge.yaml new file mode 100644 index 0000000..3f0fbbc --- /dev/null +++ b/kubespray/inventory/sample/patches/kube-controller-manager+merge.yaml @@ -0,0 +1,8 @@ +--- +apiVersion: v1 +kind: Pod +metadata: + name: kube-controller-manager + annotations: + prometheus.io/scrape: 'true' + prometheus.io/port: '10257' diff --git a/kubespray/inventory/sample/patches/kube-scheduler+merge.yaml b/kubespray/inventory/sample/patches/kube-scheduler+merge.yaml new file mode 100644 index 0000000..00f4572 --- /dev/null +++ b/kubespray/inventory/sample/patches/kube-scheduler+merge.yaml @@ -0,0 +1,8 @@ +--- +apiVersion: v1 +kind: Pod +metadata: + name: kube-scheduler + annotations: + prometheus.io/scrape: 'true' + prometheus.io/port: '10259' diff --git a/kubespray/library/kube.py b/kubespray/library/kube.py new file mode 120000 index 0000000..e5d33a6 --- /dev/null +++ b/kubespray/library/kube.py @@ -0,0 +1 @@ +../plugins/modules/kube.py \ No newline at end of file diff --git a/kubespray/logo/LICENSE b/kubespray/logo/LICENSE new file mode 100644 index 0000000..8f2aa43 --- /dev/null +++ b/kubespray/logo/LICENSE @@ -0,0 +1 @@ +# The Kubespray logo files are licensed under a choice of either Apache-2.0 or CC-BY-4.0 (Creative Commons Attribution 4.0 International). diff --git a/kubespray/logo/OWNERS b/kubespray/logo/OWNERS new file mode 100644 index 0000000..52acd54 --- /dev/null +++ b/kubespray/logo/OWNERS @@ -0,0 +1,4 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +approvers: + - thomeced diff --git a/kubespray/logo/logo-clear.png b/kubespray/logo/logo-clear.png new file mode 100644 index 0000000000000000000000000000000000000000..3ce32f6e33fecaf4c0392a88bfc7e0892fcd7552 GIT binary patch literal 4679 zcmZ`-byO2@*JjiRseu9#Dk0^FAq~RlW}wm?3c~0{kPQ$i1vW~;;YdNcdr~sG1Qa9$ z>5hTa%lrQQefOSwpL6cH_qqSw^Lu_VFntXgs=HJ~L_{=NnyN-On12)T6y!IVskM_k;h()`cZ)M%?Klf7hI; z*vwokGP{9su4`q9%W7L-T&3ijJ?`**RU{`+`q$mRZ_5jsRg)QAoN27)7pTDo#15Z1 z>ww>=;o$x-rGNz0MiW2RW=>nVZve*@cmJB&Z1sQ-sthjXr$g(KHfv%;aAv;5dw*PW zcGzQ9jh?*|@Y$NC4L?PpbcvBDTuW`$DdmEOlao|`sw;|_{(zz3IZjFv3 zfw+-+YY?bQcrqtNj12&;QBI8RWJry2aQ9^&lH}AaBn$^H5@W~z5K&C^*z)je9SEkl zOA0ZlBa?YN?3>b#$xdk*&^%v~zQl}|Ym)ImwLWX}>*=|%%;mcUrqLh4kD}i=6QbYP zQ%7QK|8gWCMiN+>v2rQ2+9$GexwpumZhO&JcxE2j9f0Yl>iV}r z!7XWXKgEi?+|OY>(}Yuu<9SJzn}2tn#xR3is;X+w;3oeah*cgO+r0^yW!rHfbl;)) z@YeO*W7aPWiSuIMr+W~;hi^+lEC-lBA{Nr+* zlv^7umY(*b^f6cXK8rK3G?6=>UB?^0>D+hb>GL;&6Ay}?t8!%bQDJu7Zi^3vOqb*C zxvjK_oT^<_hw7gmK&EN{-XFk$EW0m+hL9$j;&>Kd`FZIc?Qxu~)6<)=jA0-=Lj>cG+nJKO#Cxv&>_T8 z7u?TT3lf~Dj;Hpe9sQ=zJxFQ^NN_hMy&sj_#al1ru_>#_?h##jW|+_FPo1YF?6-p zeu)7_4I?;Mn?)|Ls+vnT>#HTFV_1cyGulQ}=tJgM6L34E;~P+*s~q=jbfrV3C9|%m zM^|6m^!Q#WUvV7*nx?_gY&zB@um6{#COdmk;R$p5UXsGUyiXCANcT;oH>mpC#)*1L z++6H}Y5-jFP}(iU-9T^4^!-IdEmGm@#}x4m!_F8u*D2G%M7{>t+U=;84)w=zXYOu1;&6L78 zLs3DNX51v-t0(+pC0j$TvNdOk`rz&rC+ZDu{E9|eT4Uk#Us+UTi`F-6pO`Wdjhn+r?c2|TzlTbj-N|+REUU8XHz(}VCEXvq zSt&0ASMz4|bpcMZt5>`$|LyDI+~u+LCgrVJMWJ&ZPJ0zMTjfPToPs|}qx1Dj11>gY z2}1dakvc%2A~GC@a(hG;mp|7|8bQ$S38k5s|pPX~|{o4zu!eSUw+9wwk}_ zP6WJ`Z10d{U96tHJQ=nlD3fHur0vhNm_*IXWBsf@3RKVQ0|I^HR1h2lUDGPx9Um3C zolyQw7+424T;XgKIcms~6mT)d8`Fk7zKt%&aC6gXpAcqumO4u0e>Q8-%qaHS-oC)L z)*gafY=kK&&XsqINxwg>BdsP~X|^xPl196c+=*Qudksy7y7&eulKSAX+U2zxl16Re zZPtFdx|}a8@5|bRoRr)1HRyg)Ax3YKVeSub<9t3q8%yxpL6|Pw(1MIG0RPoiMLApk zsYZ6osW5`8i9CS0s!@kmQUMAGRm@zq5sR^k^_!-D{0YHX8Sgpv%5*L)4&bR}L+-2% zTR(D-9L{lVS*;A)=(mU~lTA!G$UxHhNWIlTZ-~Y8RV@sFjYn92z9YQ*xw&GxBnx)y z|NFaQWO`#t#4?dFQ)c9e_8KI$TZHj1*HWx*rS>ko~-PxRGPU?;O(a(zWhUlRmM1j>%hg zGv*PUG785Yqwcz;%vuj884JF7a!Ho|TBE5ito*fM6V{K3+>|kQtGH-u_~tDzeCFV8 zKk-`^Wx!%>NAUZuJ#%;aqgl)f@H4>0RDyc5_G`T`b99xCqQIC1+wyDmRz>Fy&_rHL zamD9@F1G$1$JM1yM2dq)?Dp3J_bKBSU_CEzbcd>$bY9g7m+(spJk>~uezHM>d6`Cn z+&xdQNGy(}-aJfzcNTkTQje>OV8o}lSy>wTR4<#zP88abSy}?dU*rH|#YE$=F}(EO z!yyIp9mi61kc_dP`D+6>W>@_Qoq2Zx+3u>@45~v~T8xmSAx8R4=_GF@U6tSPLquK5 znk!EL7h~0Q?gZoZknbBdxEeuVSOrDtq)L-2(h=r924z(;N0FM7*!$%!+e+!5dxyM9 zdX;ZbNG{dfmWYv0LU4`51;HQhbd>qLAd_!NG2%2=hpE?UdYm~I;j$)$=~+15>Bn7~ z)5i0ad%O^U3g{6K;*Zb27##*>WKjoCcw56~kdOlBmvj=Q5Q~ia=!n(GX5owuP1uVv zxlwgCBa4s%v`CsokdocB5(KX~^ibR@bYoRCDfv6{-3cwA`085zu_-0T> z3fMA?FngGxSW~LPI$j=oB!O)(4z4&~uuOf)sE&m6%j$Xr`4r*Id7B~@kJ?hRqYGHj zMVYu_os7qE`IHRRDlA#PNX#=awAD}|?qa~RGaaGTve~qy@K`NZmG3+1=sAsCiQL0Zp@EgrP&Z^UW`GQx^QXe&%CX7y5yZ5I+E4a|wD=Kvi$vh7 za)tA)p__LC4}VIyw@yB)k)7RaNeiya&--?Em_VtGgiyn(wprqqw}Vi3hP+)lvc_ND zObY<_t84oCs81+LJaM?u7u%3DXOB+~tnYzlL%$}$ z**7dyPHw;bQh~GgI{#Yufpvh4Hzi)uua`!{Unm}bnVc@{)otz86(3=~9%fFQ9M+tH%%F92 z-b-k>@or|i7_$-xO(j^66q8N;o~4KU%La=Mvv1xTzv@B??F1Fdf2Dy%e3pf&F1FKg zMoiyho{A*@`tbqtj2Ybm{{z)tA-v9{zv=CV@cvO2alaz)bdh!NS} z#Z067K62iaJkSie(AF;Jzjt>Lt)q6P>w;wAvu{<30}C^tXKD(iiSx^MHIb0|f8Z?< zMRTJi&CG^47w4p;kX#zGyM43qokf8=NjRC%`xC^M-1kI5=(F)b=@s4HQODZ>L7%w^ zb^WYgqw55R>4}(po*o7zj>{;T0KAb0Jm}u%#1A*B9ym>M2M&d7p>ScNm!3n#wxW(y z$>A~{IwIJ9A7IB``|Hl>e7&u=%1H0RSj^T(X)isqVwhoM1t13-ig3(j zE8rM@A3ZkM%89U1m6|X4N>_BXEX(s_Azt0D&w0^&ejGK{je5nKy}3F``@si%PnU`C zDNV|tTw3!vm^xFz$ej;Wv_pSuRqD>+>BnW(ELgY;ro5@FA~KYY^Pjq2c+6LmTR-^J z-`AYDGAs`;?k1#|H0|jJxYjw)8i{ zOx_{Y&h`=&Cs@*@<6b*`lu0IdsCdA#B zlKK2yse8)Gya{VmR+#o7EAvRB(ko#2MTR8eNO&ZvYfESI77{Y-pNER7yoz>~cS`Hs z=pufNgwSm`w*P_SVV*{gk8Q~4UJF1o#H}nAPbNS?@}A0P#PX-NN%M__umRk+^!?VP z>s7Rt79UAv0>bGi@TKd*Hj_)v)^670gH`h@O_F5BISc zj)p85ni!-90~CU2C|0HE6O>u^i>PH=b;om>!UmkU{wUQ@ti=9LnWMG=NWZz6lL?<& zE3XcULbmWO#lk-_=O^jW7PgYhq&KCuBjO@<4BG@XZ_ospKm$C6mxgFM$4$j-w!+dz8h4iFt|_ABnTY;lUnxn=;SL&G}yK=6jfSOWLMMaA*iA zgOUSZI$MGG>7d+E|9r2BHmubFuS1eM8W@joAfQ}?yLgN*!kvZUe)uVH%Ym}VHZX`L zV7Lb#<3@VN2@v0(E3|t^!7W+;nN3Ie|FmS2GsQH$()M9inAXkD5RsOezG|7WE%HB0 C;MWQO literal 0 HcmV?d00001 diff --git a/kubespray/logo/logo-clear.svg b/kubespray/logo/logo-clear.svg new file mode 100644 index 0000000..7d60232 --- /dev/null +++ b/kubespray/logo/logo-clear.svg @@ -0,0 +1,80 @@ + + image/svg+xml + + + + + + + + + + + + + + + + background + + + + Layer 1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/kubespray/logo/logo-dark.png b/kubespray/logo/logo-dark.png new file mode 100644 index 0000000000000000000000000000000000000000..5fc3660668ee5af3553353406aa55aa1792688f0 GIT binary patch literal 6360 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 a5aaih3T4^iyXd3<0000 + image/svg+xml + + + + + + + + + + + + + + + + background + + + + Layer 1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/kubespray/logo/logo-text-clear.png b/kubespray/logo/logo-text-clear.png new file mode 100644 index 0000000000000000000000000000000000000000..b8412407d3c3138e8f408514bdfab2fb3a2f8ca1 GIT binary patch literal 13074 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 literal 0 HcmV?d00001 diff --git a/kubespray/logo/logo-text-clear.svg b/kubespray/logo/logo-text-clear.svg new file mode 100644 index 0000000..b1029de --- /dev/null +++ b/kubespray/logo/logo-text-clear.svg @@ -0,0 +1,107 @@ + + image/svg+xml + + + + + + + + + + + + + + + + background + + + + Layer 1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/kubespray/logo/logo-text-dark.png b/kubespray/logo/logo-text-dark.png new file mode 100644 index 0000000000000000000000000000000000000000..1871c0fb37749de7df02d157df3c2fc5bbfda965 GIT binary patch literal 13384 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 literal 0 HcmV?d00001 diff --git a/kubespray/logo/logo-text-dark.svg b/kubespray/logo/logo-text-dark.svg new file mode 100644 index 0000000..52bdb4e --- /dev/null +++ b/kubespray/logo/logo-text-dark.svg @@ -0,0 +1,110 @@ + + image/svg+xml + + + + + + + + + + + + + + + + background + + + + Layer 1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/kubespray/logo/logo-text-mixed.png b/kubespray/logo/logo-text-mixed.png new file mode 100644 index 0000000000000000000000000000000000000000..a4b3b39d8a7df9fd81c53856ba70e8f3be8bffe8 GIT binary patch literal 16076 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 + image/svg+xml + + + + + + + + + + + + + + + + background + + + + Layer 1 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/kubespray/logo/logos.pdf b/kubespray/logo/logos.pdf new file mode 100644 index 0000000000000000000000000000000000000000..ed7a1f5f84cb7cbf18afd9fb2027a743774607b7 GIT binary patch literal 288304 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%0Ak!VKl1~IN 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

    qiwXBA1mO$A78XhqCbuXrI#z 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ȳI}eKZ!~ujn{4L!qHbKQ(iunnwVdhN__YCW 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 bb9FO!ar>)xaTb9=a7r>>US&-rCE5P~0p|_* literal 0 HcmV?d00001 diff --git a/kubespray/logo/usage_guidelines.md b/kubespray/logo/usage_guidelines.md new file mode 100644 index 0000000..9a08123 --- /dev/null +++ b/kubespray/logo/usage_guidelines.md @@ -0,0 +1,16 @@ +# Kubernetes Branding Guidelines + +These guidelines provide you with guidance for using the Kubespray logo. +All artwork is made available under the Linux Foundation trademark usage +[guidelines](https://www.linuxfoundation.org/trademark-usage/). This text from +those guidelines, and the correct and incorrect usage examples, are particularly +helpful: +>Certain marks of The Linux Foundation have been created to enable you to +>communicate compatibility or interoperability of software or products. In +>addition to the requirement that any use of a mark to make an assertion of +>compatibility must, of course, be accurate, the use of these marks must +>avoid confusion regarding The Linux Foundation’s association with the +>product. The use of the mark cannot imply that The Linux Foundation or +>its projects are sponsoring or endorsing the product. + +Additionally, permission is granted to modify the Kubespray mark for non-commercial uses such as t-shirts and stickers. diff --git a/kubespray/meta/runtime.yml b/kubespray/meta/runtime.yml new file mode 100644 index 0000000..be99ccf --- /dev/null +++ b/kubespray/meta/runtime.yml @@ -0,0 +1,2 @@ +--- +requires_ansible: '>=2.14.0' diff --git a/kubespray/pipeline.Dockerfile b/kubespray/pipeline.Dockerfile new file mode 100644 index 0000000..eb4dcf6 --- /dev/null +++ b/kubespray/pipeline.Dockerfile @@ -0,0 +1,58 @@ +# Use imutable image tags rather than mutable tags (like ubuntu:22.04) +FROM ubuntu:jammy-20230308 +# Some tools like yamllint need this +# Pip needs this as well at the moment to install ansible +# (and potentially other packages) +# See: https://github.com/pypa/pip/issues/10219 +ENV VAGRANT_VERSION=2.3.4 \ + VAGRANT_DEFAULT_PROVIDER=libvirt \ + VAGRANT_ANSIBLE_TAGS=facts \ + LANG=C.UTF-8 \ + DEBIAN_FRONTEND=noninteractive \ + PYTHONDONTWRITEBYTECODE=1 + +RUN apt update -q \ + && apt install -yq \ + libssl-dev \ + python3-dev \ + python3-pip \ + sshpass \ + apt-transport-https \ + jq \ + moreutils \ + libvirt-dev \ + openssh-client \ + rsync \ + git \ + ca-certificates \ + curl \ + gnupg2 \ + software-properties-common \ + unzip \ + libvirt-clients \ + && curl -fsSL https://download.docker.com/linux/ubuntu/gpg | apt-key add - \ + && add-apt-repository "deb [arch=$(dpkg --print-architecture)] https://download.docker.com/linux/ubuntu $(lsb_release -cs) stable" \ + && apt update -q \ + && apt install --no-install-recommends -yq docker-ce \ + && apt autoremove -yqq --purge && apt clean && rm -rf /var/lib/apt/lists/* /var/log/* + +WORKDIR /kubespray + +RUN --mount=type=bind,target=./requirements.txt,src=./requirements.txt \ + --mount=type=bind,target=./tests/requirements.txt,src=./tests/requirements.txt \ + --mount=type=bind,target=./roles/kubespray-defaults/defaults/main.yaml,src=./roles/kubespray-defaults/defaults/main.yaml \ + update-alternatives --install /usr/bin/python python /usr/bin/python3 1 \ + && pip install --no-compile --no-cache-dir pip -U \ + && pip install --no-compile --no-cache-dir -r tests/requirements.txt \ + && KUBE_VERSION=$(sed -n 's/^kube_version: //p' roles/kubespray-defaults/defaults/main.yaml) \ + && curl -L https://dl.k8s.io/release/$KUBE_VERSION/bin/linux/$(dpkg --print-architecture)/kubectl -o /usr/local/bin/kubectl \ + && echo $(curl -L https://dl.k8s.io/release/$KUBE_VERSION/bin/linux/$(dpkg --print-architecture)/kubectl.sha256) /usr/local/bin/kubectl | sha256sum --check \ + && chmod a+x /usr/local/bin/kubectl \ + # Install Vagrant + && curl -LO https://releases.hashicorp.com/vagrant/${VAGRANT_VERSION}/vagrant_${VAGRANT_VERSION}-1_$(dpkg --print-architecture).deb \ + && dpkg -i vagrant_${VAGRANT_VERSION}-1_$(dpkg --print-architecture).deb \ + && rm vagrant_${VAGRANT_VERSION}-1_$(dpkg --print-architecture).deb \ + && vagrant plugin install vagrant-libvirt \ + # Install Kubernetes collections + && pip install --no-compile --no-cache-dir kubernetes \ + && ansible-galaxy collection install kubernetes.core diff --git a/kubespray/playbooks/ansible_version.yml b/kubespray/playbooks/ansible_version.yml new file mode 100644 index 0000000..840749a --- /dev/null +++ b/kubespray/playbooks/ansible_version.yml @@ -0,0 +1,34 @@ +--- +- name: Check Ansible version + hosts: localhost + gather_facts: false + become: no + vars: + minimal_ansible_version: 2.14.0 + maximal_ansible_version: 2.15.0 + ansible_connection: local + tags: always + tasks: + - name: "Check {{ minimal_ansible_version }} <= Ansible version < {{ maximal_ansible_version }}" + assert: + msg: "Ansible must be between {{ minimal_ansible_version }} and {{ maximal_ansible_version }} exclusive - you have {{ ansible_version.string }}" + that: + - ansible_version.string is version(minimal_ansible_version, ">=") + - ansible_version.string is version(maximal_ansible_version, "<") + tags: + - check + + - name: "Check that python netaddr is installed" + assert: + msg: "Python netaddr is not present" + that: "'127.0.0.1' | ipaddr" + tags: + - check + + # CentOS 7 provides too old jinja version + - name: "Check that jinja is not too old (install via pip)" + assert: + msg: "Your Jinja version is too old, install via pip" + that: "{% set test %}It works{% endset %}{{ test == 'It works' }}" + tags: + - check diff --git a/kubespray/playbooks/cluster.yml b/kubespray/playbooks/cluster.yml new file mode 100644 index 0000000..991ae23 --- /dev/null +++ b/kubespray/playbooks/cluster.yml @@ -0,0 +1,133 @@ +--- +- name: Check ansible version + import_playbook: ansible_version.yml + +- name: Ensure compatibility with old groups + import_playbook: legacy_groups.yml + +- name: Install bastion ssh config + hosts: bastion[0] + gather_facts: False + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: bastion-ssh-config, tags: ["localhost", "bastion"] } + +- name: Gather facts + tags: always + import_playbook: facts.yml + +- name: Prepare for etcd install + hosts: k8s_cluster:etcd + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes/preinstall, tags: preinstall } + - { role: "container-engine", tags: "container-engine", when: deploy_container_engine } + - { role: download, tags: download, when: "not skip_downloads" } + +- name: Install etcd + hosts: etcd:kube_control_plane + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - role: etcd + tags: etcd + vars: + etcd_cluster_setup: true + etcd_events_cluster_setup: "{{ etcd_events_cluster_enabled }}" + when: etcd_deployment_type != "kubeadm" + +- name: Install etcd certs on nodes if required + hosts: k8s_cluster + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - role: etcd + tags: etcd + vars: + etcd_cluster_setup: false + etcd_events_cluster_setup: false + when: + - etcd_deployment_type != "kubeadm" + - kube_network_plugin in ["calico", "flannel", "canal", "cilium"] or cilium_deploy_additionally | default(false) | bool + - kube_network_plugin != "calico" or calico_datastore == "etcd" + +- name: Install Kubernetes nodes + hosts: k8s_cluster + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes/node, tags: node } + +- name: Install the control plane + hosts: kube_control_plane + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes/control-plane, tags: master } + - { role: kubernetes/client, tags: client } + - { role: kubernetes-apps/cluster_roles, tags: cluster-roles } + +- name: Invoke kubeadm and install a CNI + hosts: k8s_cluster + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes/kubeadm, tags: kubeadm} + - { role: kubernetes/node-label, tags: node-label } + - { role: network_plugin, tags: network } + - { role: kubernetes-apps/kubelet-csr-approver, tags: kubelet-csr-approver } + +- name: Install Calico Route Reflector + hosts: calico_rr + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: network_plugin/calico/rr, tags: ['network', 'calico_rr'] } + +- name: Patch Kubernetes for Windows + hosts: kube_control_plane[0] + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: win_nodes/kubernetes_patch, tags: ["master", "win_nodes"] } + +- name: Install Kubernetes apps + hosts: kube_control_plane + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes-apps/external_cloud_controller, tags: external-cloud-controller } + - { role: kubernetes-apps/network_plugin, tags: network } + - { role: kubernetes-apps/policy_controller, tags: policy-controller } + - { role: kubernetes-apps/ingress_controller, tags: ingress-controller } + - { role: kubernetes-apps/external_provisioner, tags: external-provisioner } + - { role: kubernetes-apps, tags: apps } + +- name: Apply resolv.conf changes now that cluster DNS is up + hosts: k8s_cluster + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes/preinstall, when: "dns_mode != 'none' and resolvconf_mode == 'host_resolvconf'", tags: resolvconf, dns_late: true } diff --git a/kubespray/playbooks/facts.yml b/kubespray/playbooks/facts.yml new file mode 100644 index 0000000..77823ac --- /dev/null +++ b/kubespray/playbooks/facts.yml @@ -0,0 +1,41 @@ +--- +- name: Bootstrap hosts for Ansible + hosts: k8s_cluster:etcd:calico_rr + strategy: linear + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + gather_facts: false + environment: "{{ proxy_disable_env }}" + vars: + # Need to disable pipelining for bootstrap-os as some systems have requiretty in sudoers set, which makes pipelining + # fail. bootstrap-os fixes this on these systems, so in later plays it can be enabled. + ansible_ssh_pipelining: false + roles: + - { role: kubespray-defaults } + - { role: bootstrap-os, tags: bootstrap-os} + +- name: Gather facts + hosts: k8s_cluster:etcd:calico_rr + gather_facts: False + tags: always + tasks: + - name: Gather minimal facts + setup: + gather_subset: '!all' + + # filter match the following variables: + # ansible_default_ipv4 + # ansible_default_ipv6 + # ansible_all_ipv4_addresses + # ansible_all_ipv6_addresses + - name: Gather necessary facts (network) + setup: + gather_subset: '!all,!min,network' + filter: "ansible_*_ipv[46]*" + + # filter match the following variables: + # ansible_memtotal_mb + # ansible_swaptotal_mb + - name: Gather necessary facts (hardware) + setup: + gather_subset: '!all,!min,hardware' + filter: "ansible_*total_mb" diff --git a/kubespray/playbooks/legacy_groups.yml b/kubespray/playbooks/legacy_groups.yml new file mode 100644 index 0000000..643032f --- /dev/null +++ b/kubespray/playbooks/legacy_groups.yml @@ -0,0 +1,47 @@ +--- +# This is an inventory compatibility playbook to ensure we keep compatibility with old style group names + +- name: Add kube-master nodes to kube_control_plane + hosts: kube-master + gather_facts: false + tags: always + tasks: + - name: Add nodes to kube_control_plane group + group_by: + key: 'kube_control_plane' + +- name: Add kube-node nodes to kube_node + hosts: kube-node + gather_facts: false + tags: always + tasks: + - name: Add nodes to kube_node group + group_by: + key: 'kube_node' + +- name: Add k8s-cluster nodes to k8s_cluster + hosts: k8s-cluster + gather_facts: false + tags: always + tasks: + - name: Add nodes to k8s_cluster group + group_by: + key: 'k8s_cluster' + +- name: Add calico-rr nodes to calico_rr + hosts: calico-rr + gather_facts: false + tags: always + tasks: + - name: Add nodes to calico_rr group + group_by: + key: 'calico_rr' + +- name: Add no-floating nodes to no_floating + hosts: no-floating + gather_facts: false + tags: always + tasks: + - name: Add nodes to no-floating group + group_by: + key: 'no_floating' diff --git a/kubespray/playbooks/recover_control_plane.yml b/kubespray/playbooks/recover_control_plane.yml new file mode 100644 index 0000000..d2bb574 --- /dev/null +++ b/kubespray/playbooks/recover_control_plane.yml @@ -0,0 +1,39 @@ +--- +- name: Check ansible version + import_playbook: ansible_version.yml + +- name: Ensure compatibility with old groups + import_playbook: legacy_groups.yml + +- name: Install bastion ssh config + hosts: bastion[0] + gather_facts: False + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults} + - { role: bastion-ssh-config, tags: ["localhost", "bastion"]} + +- name: Recover etcd + hosts: etcd[0] + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults} + - role: recover_control_plane/etcd + when: etcd_deployment_type != "kubeadm" + +- name: Recover control plane + hosts: kube_control_plane[0] + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults} + - { role: recover_control_plane/control-plane } + +- name: Apply whole cluster install + import_playbook: cluster.yml + +- name: Perform post recover tasks + hosts: kube_control_plane + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults} + - { role: recover_control_plane/post-recover } diff --git a/kubespray/playbooks/remove_node.yml b/kubespray/playbooks/remove_node.yml new file mode 100644 index 0000000..63df859 --- /dev/null +++ b/kubespray/playbooks/remove_node.yml @@ -0,0 +1,54 @@ +--- +- name: Check ansible version + import_playbook: ansible_version.yml + +- name: Ensure compatibility with old groups + import_playbook: legacy_groups.yml + +- name: Install bastion ssh config + hosts: bastion[0] + gather_facts: False + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: bastion-ssh-config, tags: ["localhost", "bastion"] } + +- name: Confirm node removal + hosts: "{{ node | default('etcd:k8s_cluster:calico_rr') }}" + gather_facts: no + tasks: + - name: Confirm Execution + pause: + prompt: "Are you sure you want to delete nodes state? Type 'yes' to delete nodes." + register: pause_result + run_once: True + when: + - not (skip_confirmation | default(false) | bool) + + - name: Fail if user does not confirm deletion + fail: + msg: "Delete nodes confirmation failed" + when: pause_result.user_input | default('yes') != 'yes' + +- name: Gather facts + import_playbook: facts.yml + when: reset_nodes | default(True) | bool + +- name: Reset node + hosts: "{{ node | default('kube_node') }}" + gather_facts: no + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults, when: reset_nodes | default(True) | bool } + - { role: remove-node/pre-remove, tags: pre-remove } + - { role: remove-node/remove-etcd-node } + - { role: reset, tags: reset, when: reset_nodes | default(True) | bool } + +# Currently cannot remove first master or etcd +- name: Post node removal + hosts: "{{ node | default('kube_control_plane[1:]:etcd[1:]') }}" + gather_facts: no + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults, when: reset_nodes | default(True) | bool } + - { role: remove-node/post-remove, tags: post-remove } diff --git a/kubespray/playbooks/reset.yml b/kubespray/playbooks/reset.yml new file mode 100644 index 0000000..0b4312f --- /dev/null +++ b/kubespray/playbooks/reset.yml @@ -0,0 +1,46 @@ +--- +- name: Check ansible version + import_playbook: ansible_version.yml + +- name: Ensure compatibility with old groups + import_playbook: legacy_groups.yml + +- name: Install bastion ssh config + hosts: bastion[0] + gather_facts: False + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults} + - { role: bastion-ssh-config, tags: ["localhost", "bastion"]} + +- name: Gather facts + import_playbook: facts.yml + +- name: Reset cluster + hosts: etcd:k8s_cluster:calico_rr + gather_facts: False + pre_tasks: + - name: Reset Confirmation + pause: + prompt: "Are you sure you want to reset cluster state? Type 'yes' to reset your cluster." + register: reset_confirmation_prompt + run_once: True + when: + - not (skip_confirmation | default(false) | bool) + - reset_confirmation is not defined + + - name: Check confirmation + fail: + msg: "Reset confirmation failed" + when: + - not reset_confirmation | default(false) | bool + - not reset_confirmation_prompt.user_input | default("") == "yes" + + - name: Gather information about installed services + service_facts: + + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults} + - { role: kubernetes/preinstall, when: "dns_mode != 'none' and resolvconf_mode == 'host_resolvconf'", tags: resolvconf, dns_early: true } + - { role: reset, tags: reset } diff --git a/kubespray/playbooks/scale.yml b/kubespray/playbooks/scale.yml new file mode 100644 index 0000000..007a656 --- /dev/null +++ b/kubespray/playbooks/scale.yml @@ -0,0 +1,115 @@ +--- +- name: Check ansible version + import_playbook: ansible_version.yml + +- name: Ensure compatibility with old groups + import_playbook: legacy_groups.yml + +- name: Install bastion ssh config + hosts: bastion[0] + gather_facts: False + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: bastion-ssh-config, tags: ["localhost", "bastion"] } + +- name: Gather facts + tags: always + import_playbook: facts.yml + +- name: Generate the etcd certificates beforehand + hosts: etcd:kube_control_plane + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - role: etcd + tags: etcd + vars: + etcd_cluster_setup: false + etcd_events_cluster_setup: false + when: + - etcd_deployment_type != "kubeadm" + - kube_network_plugin in ["calico", "flannel", "canal", "cilium"] or cilium_deploy_additionally | default(false) | bool + - kube_network_plugin != "calico" or calico_datastore == "etcd" + +- name: Download images to ansible host cache via first kube_control_plane node + hosts: kube_control_plane[0] + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults, when: "not skip_downloads and download_run_once and not download_localhost" } + - { role: kubernetes/preinstall, tags: preinstall, when: "not skip_downloads and download_run_once and not download_localhost" } + - { role: download, tags: download, when: "not skip_downloads and download_run_once and not download_localhost" } + +- name: Target only workers to get kubelet installed and checking in on any new nodes(engine) + hosts: kube_node + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes/preinstall, tags: preinstall } + - { role: container-engine, tags: "container-engine", when: deploy_container_engine } + - { role: download, tags: download, when: "not skip_downloads" } + - role: etcd + tags: etcd + vars: + etcd_cluster_setup: false + when: + - etcd_deployment_type != "kubeadm" + - kube_network_plugin in ["calico", "flannel", "canal", "cilium"] or cilium_deploy_additionally | default(false) | bool + - kube_network_plugin != "calico" or calico_datastore == "etcd" + +- name: Target only workers to get kubelet installed and checking in on any new nodes(node) + hosts: kube_node + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes/node, tags: node } + +- name: Upload control plane certs and retrieve encryption key + hosts: kube_control_plane | first + environment: "{{ proxy_disable_env }}" + gather_facts: False + tags: kubeadm + roles: + - { role: kubespray-defaults } + tasks: + - name: Upload control plane certificates + command: >- + {{ bin_dir }}/kubeadm init phase + --config {{ kube_config_dir }}/kubeadm-config.yaml + upload-certs + --upload-certs + environment: "{{ proxy_disable_env }}" + register: kubeadm_upload_cert + changed_when: false + - name: Set fact 'kubeadm_certificate_key' for later use + set_fact: + kubeadm_certificate_key: "{{ kubeadm_upload_cert.stdout_lines[-1] | trim }}" + when: kubeadm_certificate_key is not defined + +- name: Target only workers to get kubelet installed and checking in on any new nodes(network) + hosts: kube_node + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes/kubeadm, tags: kubeadm } + - { role: kubernetes/node-label, tags: node-label } + - { role: network_plugin, tags: network } + +- name: Apply resolv.conf changes now that cluster DNS is up + hosts: k8s_cluster + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes/preinstall, when: "dns_mode != 'none' and resolvconf_mode == 'host_resolvconf'", tags: resolvconf, dns_late: true } diff --git a/kubespray/playbooks/upgrade_cluster.yml b/kubespray/playbooks/upgrade_cluster.yml new file mode 100644 index 0000000..d546998 --- /dev/null +++ b/kubespray/playbooks/upgrade_cluster.yml @@ -0,0 +1,168 @@ +--- +- name: Check ansible version + import_playbook: ansible_version.yml + +- name: Ensure compatibility with old groups + import_playbook: legacy_groups.yml + +- name: Install bastion ssh config + hosts: bastion[0] + gather_facts: False + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: bastion-ssh-config, tags: ["localhost", "bastion"] } + +- name: Gather facts + tags: always + import_playbook: facts.yml + +- name: Download images to ansible host cache via first kube_control_plane node + hosts: kube_control_plane[0] + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults, when: "not skip_downloads and download_run_once and not download_localhost"} + - { role: kubernetes/preinstall, tags: preinstall, when: "not skip_downloads and download_run_once and not download_localhost" } + - { role: download, tags: download, when: "not skip_downloads and download_run_once and not download_localhost" } + +- name: Prepare nodes for upgrade + hosts: k8s_cluster:etcd:calico_rr + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes/preinstall, tags: preinstall } + - { role: download, tags: download, when: "not skip_downloads" } + +- name: Upgrade container engine on non-cluster nodes + hosts: etcd:calico_rr:!k8s_cluster + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + serial: "{{ serial | default('20%') }}" + roles: + - { role: kubespray-defaults } + - { role: container-engine, tags: "container-engine", when: deploy_container_engine } + +- name: Install etcd + hosts: etcd:kube_control_plane + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - role: etcd + tags: etcd + vars: + etcd_cluster_setup: true + etcd_events_cluster_setup: "{{ etcd_events_cluster_enabled }}" + when: etcd_deployment_type != "kubeadm" + +- name: Install etcd certs on nodes if required + hosts: k8s_cluster + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - role: etcd + tags: etcd + vars: + etcd_cluster_setup: false + etcd_events_cluster_setup: false + when: + - etcd_deployment_type != "kubeadm" + - kube_network_plugin in ["calico", "flannel", "canal", "cilium"] or cilium_deploy_additionally | default(false) | bool + - kube_network_plugin != "calico" or calico_datastore == "etcd" + +- name: Handle upgrades to master components first to maintain backwards compat. + gather_facts: False + hosts: kube_control_plane + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + serial: 1 + roles: + - { role: kubespray-defaults } + - { role: upgrade/pre-upgrade, tags: pre-upgrade } + - { role: upgrade/system-upgrade, tags: system-upgrade } + - { role: download, tags: download, when: "system_upgrade and system_upgrade_reboot != 'never' and not skip_downloads" } + - { role: kubernetes-apps/kubelet-csr-approver, tags: kubelet-csr-approver } + - { role: container-engine, tags: "container-engine", when: deploy_container_engine } + - { role: kubernetes/node, tags: node } + - { role: kubernetes/control-plane, tags: master, upgrade_cluster_setup: true } + - { role: kubernetes/client, tags: client } + - { role: kubernetes/node-label, tags: node-label } + - { role: kubernetes-apps/cluster_roles, tags: cluster-roles } + - { role: kubernetes-apps, tags: csi-driver } + - { role: upgrade/post-upgrade, tags: post-upgrade } + +- name: Upgrade calico and external cloud provider on all masters, calico-rrs, and nodes + hosts: kube_control_plane:calico_rr:kube_node + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + serial: "{{ serial | default('20%') }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes-apps/external_cloud_controller, tags: external-cloud-controller } + - { role: network_plugin, tags: network } + - { role: kubernetes-apps/network_plugin, tags: network } + - { role: kubernetes-apps/policy_controller, tags: policy-controller } + +- name: Finally handle worker upgrades, based on given batch size + hosts: kube_node:calico_rr:!kube_control_plane + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + serial: "{{ serial | default('20%') }}" + roles: + - { role: kubespray-defaults } + - { role: upgrade/pre-upgrade, tags: pre-upgrade } + - { role: upgrade/system-upgrade, tags: system-upgrade } + - { role: download, tags: download, when: "system_upgrade and system_upgrade_reboot != 'never' and not skip_downloads" } + - { role: container-engine, tags: "container-engine", when: deploy_container_engine } + - { role: kubernetes/node, tags: node } + - { role: kubernetes/kubeadm, tags: kubeadm } + - { role: kubernetes/node-label, tags: node-label } + - { role: upgrade/post-upgrade, tags: post-upgrade } + +- name: Patch Kubernetes for Windows + hosts: kube_control_plane[0] + gather_facts: False + any_errors_fatal: true + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: win_nodes/kubernetes_patch, tags: ["master", "win_nodes"] } + +- name: Install Calico Route Reflector + hosts: calico_rr + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: network_plugin/calico/rr, tags: network } + +- name: Install Kubernetes apps + hosts: kube_control_plane + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes-apps/ingress_controller, tags: ingress-controller } + - { role: kubernetes-apps/external_provisioner, tags: external-provisioner } + - { role: kubernetes-apps, tags: apps } + +- name: Apply resolv.conf changes now that cluster DNS is up + hosts: k8s_cluster + gather_facts: False + any_errors_fatal: "{{ any_errors_fatal | default(true) }}" + environment: "{{ proxy_disable_env }}" + roles: + - { role: kubespray-defaults } + - { role: kubernetes/preinstall, when: "dns_mode != 'none' and resolvconf_mode == 'host_resolvconf'", tags: resolvconf, dns_late: true } diff --git a/kubespray/plugins/modules/kube.py b/kubespray/plugins/modules/kube.py new file mode 100644 index 0000000..4b1e4ce --- /dev/null +++ b/kubespray/plugins/modules/kube.py @@ -0,0 +1,366 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- + +DOCUMENTATION = """ +--- +module: kube +short_description: Manage Kubernetes Cluster +description: + - Create, replace, remove, and stop resources within a Kubernetes Cluster +version_added: "2.0" +options: + name: + required: false + default: null + description: + - The name associated with resource + filename: + required: false + default: null + description: + - The path and filename of the resource(s) definition file(s). + - To operate on several files this can accept a comma separated list of files or a list of files. + aliases: [ 'files', 'file', 'filenames' ] + kubectl: + required: false + default: null + description: + - The path to the kubectl bin + namespace: + required: false + default: null + description: + - The namespace associated with the resource(s) + resource: + required: false + default: null + description: + - The resource to perform an action on. pods (po), replicationControllers (rc), services (svc) + label: + required: false + default: null + description: + - The labels used to filter specific resources. + server: + required: false + default: null + description: + - The url for the API server that commands are executed against. + kubeconfig: + required: false + default: null + description: + - The path to the kubeconfig. + force: + required: false + default: false + description: + - A flag to indicate to force delete, replace, or stop. + wait: + required: false + default: false + description: + - A flag to indicate to wait for resources to be created before continuing to the next step + all: + required: false + default: false + description: + - A flag to indicate delete all, stop all, or all namespaces when checking exists. + log_level: + required: false + default: 0 + description: + - Indicates the level of verbosity of logging by kubectl. + state: + required: false + choices: ['present', 'absent', 'latest', 'reloaded', 'stopped'] + default: present + description: + - present handles checking existence or creating if definition file provided, + absent handles deleting resource(s) based on other options, + latest handles creating or updating based on existence, + reloaded handles updating resource(s) definition using definition file, + stopped handles stopping resource(s) based on other options. + recursive: + required: false + default: false + description: + - Process the directory used in -f, --filename recursively. + Useful when you want to manage related manifests organized + within the same directory. +requirements: + - kubectl +author: "Kenny Jones (@kenjones-cisco)" +""" + +EXAMPLES = """ +- name: test nginx is present + kube: name=nginx resource=rc state=present + +- name: test nginx is stopped + kube: name=nginx resource=rc state=stopped + +- name: test nginx is absent + kube: name=nginx resource=rc state=absent + +- name: test nginx is present + kube: filename=/tmp/nginx.yml + +- name: test nginx and postgresql are present + kube: files=/tmp/nginx.yml,/tmp/postgresql.yml + +- name: test nginx and postgresql are present + kube: + files: + - /tmp/nginx.yml + - /tmp/postgresql.yml +""" + + +class KubeManager(object): + + def __init__(self, module): + + self.module = module + + self.kubectl = module.params.get('kubectl') + if self.kubectl is None: + self.kubectl = module.get_bin_path('kubectl', True) + self.base_cmd = [self.kubectl] + + if module.params.get('server'): + self.base_cmd.append('--server=' + module.params.get('server')) + + if module.params.get('kubeconfig'): + self.base_cmd.append('--kubeconfig=' + module.params.get('kubeconfig')) + + if module.params.get('log_level'): + self.base_cmd.append('--v=' + str(module.params.get('log_level'))) + + if module.params.get('namespace'): + self.base_cmd.append('--namespace=' + module.params.get('namespace')) + + + self.all = module.params.get('all') + self.force = module.params.get('force') + self.wait = module.params.get('wait') + self.name = module.params.get('name') + self.filename = [f.strip() for f in module.params.get('filename') or []] + self.resource = module.params.get('resource') + self.label = module.params.get('label') + self.recursive = module.params.get('recursive') + + def _execute(self, cmd): + args = self.base_cmd + cmd + try: + rc, out, err = self.module.run_command(args) + if rc != 0: + self.module.fail_json( + msg='error running kubectl (%s) command (rc=%d), out=\'%s\', err=\'%s\'' % (' '.join(args), rc, out, err)) + except Exception as exc: + self.module.fail_json( + msg='error running kubectl (%s) command: %s' % (' '.join(args), str(exc))) + return out.splitlines() + + def _execute_nofail(self, cmd): + args = self.base_cmd + cmd + rc, out, err = self.module.run_command(args) + if rc != 0: + return None + return out.splitlines() + + def create(self, check=True, force=True): + if check and self.exists(): + return [] + + cmd = ['apply'] + + if force: + cmd.append('--force') + + if self.wait: + cmd.append('--wait') + + if self.recursive: + cmd.append('--recursive={}'.format(self.recursive)) + + if not self.filename: + self.module.fail_json(msg='filename required to create') + + cmd.append('--filename=' + ','.join(self.filename)) + + return self._execute(cmd) + + def replace(self, force=True): + + cmd = ['apply'] + + if force: + cmd.append('--force') + + if self.wait: + cmd.append('--wait') + + if self.recursive: + cmd.append('--recursive={}'.format(self.recursive)) + + if not self.filename: + self.module.fail_json(msg='filename required to reload') + + cmd.append('--filename=' + ','.join(self.filename)) + + return self._execute(cmd) + + def delete(self): + + if not self.force and not self.exists(): + return [] + + cmd = ['delete'] + + if self.filename: + cmd.append('--filename=' + ','.join(self.filename)) + if self.recursive: + cmd.append('--recursive={}'.format(self.recursive)) + else: + if not self.resource: + self.module.fail_json(msg='resource required to delete without filename') + + cmd.append(self.resource) + + if self.name: + cmd.append(self.name) + + if self.label: + cmd.append('--selector=' + self.label) + + if self.all: + cmd.append('--all') + + if self.force: + cmd.append('--ignore-not-found') + + if self.recursive: + cmd.append('--recursive={}'.format(self.recursive)) + + return self._execute(cmd) + + def exists(self): + cmd = ['get'] + + if self.filename: + cmd.append('--filename=' + ','.join(self.filename)) + if self.recursive: + cmd.append('--recursive={}'.format(self.recursive)) + else: + if not self.resource: + self.module.fail_json(msg='resource required without filename') + + cmd.append(self.resource) + + if self.name: + cmd.append(self.name) + + if self.label: + cmd.append('--selector=' + self.label) + + if self.all: + cmd.append('--all-namespaces') + + cmd.append('--no-headers') + + result = self._execute_nofail(cmd) + if not result: + return False + return True + + # TODO: This is currently unused, perhaps convert to 'scale' with a replicas param? + def stop(self): + + if not self.force and not self.exists(): + return [] + + cmd = ['stop'] + + if self.filename: + cmd.append('--filename=' + ','.join(self.filename)) + if self.recursive: + cmd.append('--recursive={}'.format(self.recursive)) + else: + if not self.resource: + self.module.fail_json(msg='resource required to stop without filename') + + cmd.append(self.resource) + + if self.name: + cmd.append(self.name) + + if self.label: + cmd.append('--selector=' + self.label) + + if self.all: + cmd.append('--all') + + if self.force: + cmd.append('--ignore-not-found') + + return self._execute(cmd) + + +def main(): + + module = AnsibleModule( + argument_spec=dict( + name=dict(), + filename=dict(type='list', aliases=['files', 'file', 'filenames']), + namespace=dict(), + resource=dict(), + label=dict(), + server=dict(), + kubeconfig=dict(), + kubectl=dict(), + force=dict(default=False, type='bool'), + wait=dict(default=False, type='bool'), + all=dict(default=False, type='bool'), + log_level=dict(default=0, type='int'), + state=dict(default='present', choices=['present', 'absent', 'latest', 'reloaded', 'stopped', 'exists']), + recursive=dict(default=False, type='bool'), + ), + mutually_exclusive=[['filename', 'list']] + ) + + changed = False + + manager = KubeManager(module) + state = module.params.get('state') + if state == 'present': + result = manager.create(check=False) + + elif state == 'absent': + result = manager.delete() + + elif state == 'reloaded': + result = manager.replace() + + elif state == 'stopped': + result = manager.stop() + + elif state == 'latest': + result = manager.replace() + + elif state == 'exists': + result = manager.exists() + module.exit_json(changed=changed, + msg='%s' % result) + + else: + module.fail_json(msg='Unrecognized state %s.' % state) + + module.exit_json(changed=changed, + msg='success: %s' % (' '.join(result)) + ) + + +from ansible.module_utils.basic import * # noqa +if __name__ == '__main__': + main() diff --git a/kubespray/recover-control-plane.yml b/kubespray/recover-control-plane.yml new file mode 100644 index 0000000..53b517c --- /dev/null +++ b/kubespray/recover-control-plane.yml @@ -0,0 +1,3 @@ +--- +- name: Recover control plane + ansible.builtin.import_playbook: playbooks/recover_control_plane.yml diff --git a/kubespray/remove-node.yml b/kubespray/remove-node.yml new file mode 100644 index 0000000..7a336c6 --- /dev/null +++ b/kubespray/remove-node.yml @@ -0,0 +1,3 @@ +--- +- name: Remove node + ansible.builtin.import_playbook: playbooks/remove_node.yml diff --git a/kubespray/requirements.txt b/kubespray/requirements.txt new file mode 100644 index 0000000..2420014 --- /dev/null +++ b/kubespray/requirements.txt @@ -0,0 +1,9 @@ +ansible==7.6.0 +cryptography==41.0.1 +jinja2==3.1.2 +jmespath==1.0.1 +MarkupSafe==2.1.3 +netaddr==0.8.0 +pbr==5.11.1 +ruamel.yaml==0.17.31 +ruamel.yaml.clib==0.2.7 diff --git a/kubespray/reset.yml b/kubespray/reset.yml new file mode 100644 index 0000000..286593d --- /dev/null +++ b/kubespray/reset.yml @@ -0,0 +1,3 @@ +--- +- name: Reset the cluster + ansible.builtin.import_playbook: playbooks/reset.yml diff --git a/kubespray/roles/adduser/defaults/main.yml b/kubespray/roles/adduser/defaults/main.yml new file mode 100644 index 0000000..df3fc2d --- /dev/null +++ b/kubespray/roles/adduser/defaults/main.yml @@ -0,0 +1,27 @@ +--- +kube_owner: kube +kube_cert_group: kube-cert +etcd_data_dir: "/var/lib/etcd" + +addusers: + etcd: + name: etcd + comment: "Etcd user" + create_home: no + system: yes + shell: /sbin/nologin + kube: + name: kube + comment: "Kubernetes user" + create_home: no + system: yes + shell: /sbin/nologin + group: "{{ kube_cert_group }}" + +adduser: + name: "{{ user.name }}" + group: "{{ user.name | default(None) }}" + comment: "{{ user.comment | default(None) }}" + shell: "{{ user.shell | default(None) }}" + system: "{{ user.system | default(None) }}" + create_home: "{{ user.create_home | default(None) }}" diff --git a/kubespray/roles/adduser/molecule/default/converge.yml b/kubespray/roles/adduser/molecule/default/converge.yml new file mode 100644 index 0000000..47ff6c7 --- /dev/null +++ b/kubespray/roles/adduser/molecule/default/converge.yml @@ -0,0 +1,10 @@ +--- +- name: Converge + hosts: all + become: true + gather_facts: false + roles: + - role: adduser + vars: + user: + name: foo diff --git a/kubespray/roles/adduser/molecule/default/molecule.yml b/kubespray/roles/adduser/molecule/default/molecule.yml new file mode 100644 index 0000000..0fb4997 --- /dev/null +++ b/kubespray/roles/adduser/molecule/default/molecule.yml @@ -0,0 +1,23 @@ +--- +role_name_check: 1 +dependency: + name: galaxy +driver: + name: vagrant + provider: + name: libvirt +platforms: + - name: adduser-01 + box: generic/ubuntu2004 + cpus: 1 + memory: 512 + provider_options: + driver: kvm +provisioner: + name: ansible + config_options: + defaults: + callbacks_enabled: profile_tasks + timeout: 120 +verifier: + name: testinfra diff --git a/kubespray/roles/adduser/molecule/default/tests/test_default.py b/kubespray/roles/adduser/molecule/default/tests/test_default.py new file mode 100644 index 0000000..7e8649d --- /dev/null +++ b/kubespray/roles/adduser/molecule/default/tests/test_default.py @@ -0,0 +1,43 @@ +import os +from pathlib import Path + +import testinfra.utils.ansible_runner +import yaml +from ansible.cli.playbook import PlaybookCLI +from ansible.playbook import Playbook + +testinfra_hosts = testinfra.utils.ansible_runner.AnsibleRunner( + os.environ["MOLECULE_INVENTORY_FILE"] +).get_hosts("all") + + +def read_playbook(playbook): + cli_args = [os.path.realpath(playbook), testinfra_hosts] + cli = PlaybookCLI(cli_args) + cli.parse() + loader, inventory, variable_manager = cli._play_prereqs() + + pb = Playbook.load(cli.args[0], variable_manager, loader) + + for play in pb.get_plays(): + yield variable_manager.get_vars(play) + + +def get_playbook(): + playbooks_path = Path(__file__).parent.parent + with open(os.path.join(playbooks_path, "molecule.yml"), "r") as yamlfile: + data = yaml.load(yamlfile, Loader=yaml.FullLoader) + if "playbooks" in data["provisioner"].keys(): + if "converge" in data["provisioner"]["playbooks"].keys(): + return data["provisioner"]["playbooks"]["converge"] + else: + return os.path.join(playbooks_path, "converge.yml") + + +def test_user(host): + for vars in read_playbook(get_playbook()): + assert host.user(vars["user"]["name"]).exists + if "group" in vars["user"].keys(): + assert host.group(vars["user"]["group"]).exists + else: + assert host.group(vars["user"]["name"]).exists diff --git a/kubespray/roles/adduser/tasks/main.yml b/kubespray/roles/adduser/tasks/main.yml new file mode 100644 index 0000000..ba5edd7 --- /dev/null +++ b/kubespray/roles/adduser/tasks/main.yml @@ -0,0 +1,16 @@ +--- +- name: User | Create User Group + group: + name: "{{ user.group | default(user.name) }}" + system: "{{ user.system | default(omit) }}" + +- name: User | Create User + user: + comment: "{{ user.comment | default(omit) }}" + create_home: "{{ user.create_home | default(omit) }}" + group: "{{ user.group | default(user.name) }}" + home: "{{ user.home | default(omit) }}" + shell: "{{ user.shell | default(omit) }}" + name: "{{ user.name }}" + system: "{{ user.system | default(omit) }}" + when: user.name != "root" diff --git a/kubespray/roles/adduser/vars/coreos.yml b/kubespray/roles/adduser/vars/coreos.yml new file mode 100644 index 0000000..5c258df --- /dev/null +++ b/kubespray/roles/adduser/vars/coreos.yml @@ -0,0 +1,8 @@ +--- +addusers: + - name: kube + comment: "Kubernetes user" + shell: /sbin/nologin + system: yes + group: "{{ kube_cert_group }}" + create_home: no diff --git a/kubespray/roles/adduser/vars/debian.yml b/kubespray/roles/adduser/vars/debian.yml new file mode 100644 index 0000000..99e5b38 --- /dev/null +++ b/kubespray/roles/adduser/vars/debian.yml @@ -0,0 +1,15 @@ +--- +addusers: + - name: etcd + comment: "Etcd user" + create_home: yes + home: "{{ etcd_data_dir }}" + system: yes + shell: /sbin/nologin + + - name: kube + comment: "Kubernetes user" + create_home: no + system: yes + shell: /sbin/nologin + group: "{{ kube_cert_group }}" diff --git a/kubespray/roles/adduser/vars/redhat.yml b/kubespray/roles/adduser/vars/redhat.yml new file mode 100644 index 0000000..99e5b38 --- /dev/null +++ b/kubespray/roles/adduser/vars/redhat.yml @@ -0,0 +1,15 @@ +--- +addusers: + - name: etcd + comment: "Etcd user" + create_home: yes + home: "{{ etcd_data_dir }}" + system: yes + shell: /sbin/nologin + + - name: kube + comment: "Kubernetes user" + create_home: no + system: yes + shell: /sbin/nologin + group: "{{ kube_cert_group }}" diff --git a/kubespray/roles/bastion-ssh-config/defaults/main.yml b/kubespray/roles/bastion-ssh-config/defaults/main.yml new file mode 100644 index 0000000..83aafed --- /dev/null +++ b/kubespray/roles/bastion-ssh-config/defaults/main.yml @@ -0,0 +1,2 @@ +--- +ssh_bastion_confing__name: ssh-bastion.conf diff --git a/kubespray/roles/bastion-ssh-config/molecule/default/converge.yml b/kubespray/roles/bastion-ssh-config/molecule/default/converge.yml new file mode 100644 index 0000000..54a6247 --- /dev/null +++ b/kubespray/roles/bastion-ssh-config/molecule/default/converge.yml @@ -0,0 +1,15 @@ +--- +- name: Converge + hosts: all + become: true + gather_facts: false + roles: + - role: bastion-ssh-config + tasks: + - name: Copy config to remote host + copy: + src: "{{ playbook_dir }}/{{ ssh_bastion_confing__name }}" + dest: "{{ ssh_bastion_confing__name }}" + owner: "{{ ansible_user }}" + group: "{{ ansible_user }}" + mode: 0644 diff --git a/kubespray/roles/bastion-ssh-config/molecule/default/molecule.yml b/kubespray/roles/bastion-ssh-config/molecule/default/molecule.yml new file mode 100644 index 0000000..11cf91c --- /dev/null +++ b/kubespray/roles/bastion-ssh-config/molecule/default/molecule.yml @@ -0,0 +1,31 @@ +--- +role_name_check: 1 +dependency: + name: galaxy +driver: + name: vagrant + provider: + name: libvirt +platforms: + - name: bastion-01 + box: generic/ubuntu2004 + cpus: 1 + memory: 512 + provider_options: + driver: kvm +provisioner: + name: ansible + config_options: + defaults: + callbacks_enabled: profile_tasks + timeout: 120 + inventory: + hosts: + all: + hosts: + children: + bastion: + hosts: + bastion-01: +verifier: + name: testinfra diff --git a/kubespray/roles/bastion-ssh-config/molecule/default/tests/test_default.py b/kubespray/roles/bastion-ssh-config/molecule/default/tests/test_default.py new file mode 100644 index 0000000..cce719d --- /dev/null +++ b/kubespray/roles/bastion-ssh-config/molecule/default/tests/test_default.py @@ -0,0 +1,40 @@ +import os +from pathlib import Path + +import testinfra.utils.ansible_runner +import yaml +from ansible.cli.playbook import PlaybookCLI +from ansible.playbook import Playbook + +testinfra_hosts = testinfra.utils.ansible_runner.AnsibleRunner( + os.environ["MOLECULE_INVENTORY_FILE"] +).get_hosts("all") + + +def read_playbook(playbook): + cli_args = [os.path.realpath(playbook), testinfra_hosts] + cli = PlaybookCLI(cli_args) + cli.parse() + loader, inventory, variable_manager = cli._play_prereqs() + + pb = Playbook.load(cli.args[0], variable_manager, loader) + + for play in pb.get_plays(): + yield variable_manager.get_vars(play) + + +def get_playbook(): + playbooks_path = Path(__file__).parent.parent + with open(os.path.join(playbooks_path, "molecule.yml"), "r") as yamlfile: + data = yaml.load(yamlfile, Loader=yaml.FullLoader) + if "playbooks" in data["provisioner"].keys(): + if "converge" in data["provisioner"]["playbooks"].keys(): + return data["provisioner"]["playbooks"]["converge"] + else: + return os.path.join(playbooks_path, "converge.yml") + + +def test_ssh_config(host): + for vars in read_playbook(get_playbook()): + assert host.file(vars["ssh_bastion_confing__name"]).exists + assert host.file(vars["ssh_bastion_confing__name"]).is_file diff --git a/kubespray/roles/bastion-ssh-config/tasks/main.yml b/kubespray/roles/bastion-ssh-config/tasks/main.yml new file mode 100644 index 0000000..920763e --- /dev/null +++ b/kubespray/roles/bastion-ssh-config/tasks/main.yml @@ -0,0 +1,22 @@ +--- +- name: Set bastion host IP and port + set_fact: + bastion_ip: "{{ hostvars[groups['bastion'][0]]['ansible_host'] | d(hostvars[groups['bastion'][0]]['ansible_ssh_host']) }}" + bastion_port: "{{ hostvars[groups['bastion'][0]]['ansible_port'] | d(hostvars[groups['bastion'][0]]['ansible_ssh_port']) | d(22) }}" + delegate_to: localhost + connection: local + +# As we are actually running on localhost, the ansible_ssh_user is your local user when you try to use it directly +# To figure out the real ssh user, we delegate this task to the bastion and store the ansible_user in real_user +- name: Store the current ansible_user in the real_user fact + set_fact: + real_user: "{{ ansible_user }}" + +- name: Create ssh bastion conf + become: false + delegate_to: localhost + connection: local + template: + src: "{{ ssh_bastion_confing__name }}.j2" + dest: "{{ playbook_dir }}/{{ ssh_bastion_confing__name }}" + mode: 0640 diff --git a/kubespray/roles/bastion-ssh-config/templates/ssh-bastion.conf.j2 b/kubespray/roles/bastion-ssh-config/templates/ssh-bastion.conf.j2 new file mode 100644 index 0000000..bd5f49c --- /dev/null +++ b/kubespray/roles/bastion-ssh-config/templates/ssh-bastion.conf.j2 @@ -0,0 +1,18 @@ +{% set vars={'hosts': ''} %} +{% set user='' %} + +{% for h in groups['all'] %} +{% if h not in groups['bastion'] %} +{% if vars.update({'hosts': vars['hosts'] + ' ' + (hostvars[h].get('ansible_ssh_host') or hostvars[h]['ansible_host'])}) %}{% endif %} +{% endif %} +{% endfor %} + +Host {{ bastion_ip }} + Hostname {{ bastion_ip }} + StrictHostKeyChecking no + ControlMaster auto + ControlPath ~/.ssh/ansible-%r@%h:%p + ControlPersist 5m + +Host {{ vars['hosts'] }} + ProxyCommand ssh -F /dev/null -o StrictHostKeyChecking=no -o UserKnownHostsFile=/dev/null -W %h:%p -p {{ bastion_port }} {{ real_user }}@{{ bastion_ip }} {% if ansible_ssh_private_key_file is defined %}-i {{ ansible_ssh_private_key_file }}{% endif %} diff --git a/kubespray/roles/bootstrap-os/defaults/main.yml b/kubespray/roles/bootstrap-os/defaults/main.yml new file mode 100644 index 0000000..9b31456 --- /dev/null +++ b/kubespray/roles/bootstrap-os/defaults/main.yml @@ -0,0 +1,32 @@ +--- +## CentOS/RHEL/AlmaLinux specific variables +# Use the fastestmirror yum plugin +centos_fastestmirror_enabled: false + +## Flatcar Container Linux specific variables +# Disable locksmithd or leave it in its current state +coreos_locksmithd_disable: false + +## Oracle Linux specific variables +# Install public repo on Oracle Linux +use_oracle_public_repo: true + +fedora_coreos_packages: + - python + - python3-libselinux + - ethtool # required in kubeadm preflight phase for verifying the environment + - ipset # required in kubeadm preflight phase for verifying the environment + - conntrack-tools # required by kube-proxy + +## General +# Set the hostname to inventory_hostname +override_system_hostname: true + +is_fedora_coreos: false + +skip_http_proxy_on_os_packages: false + +# If this is true, debug information will be displayed but +# may contain some private data, so it is recommended to set it to false +# in the production environment. +unsafe_show_logs: false diff --git a/kubespray/roles/bootstrap-os/files/bootstrap.sh b/kubespray/roles/bootstrap-os/files/bootstrap.sh new file mode 100755 index 0000000..4c479a1 --- /dev/null +++ b/kubespray/roles/bootstrap-os/files/bootstrap.sh @@ -0,0 +1,46 @@ +#!/bin/bash +set -e + +BINDIR="/opt/bin" +if [[ -e $BINDIR/.bootstrapped ]]; then + exit 0 +fi + +ARCH=$(uname -m) +case $ARCH in + "x86_64") + PYPY_ARCH=linux64 + PYPI_HASH=46818cb3d74b96b34787548343d266e2562b531ddbaf330383ba930ff1930ed5 + ;; + "aarch64") + PYPY_ARCH=aarch64 + PYPI_HASH=2e1ae193d98bc51439642a7618d521ea019f45b8fb226940f7e334c548d2b4b9 + ;; + *) + echo "Unsupported Architecture: ${ARCH}" + exit 1 +esac + +PYTHON_VERSION=3.9 +PYPY_VERSION=7.3.9 +PYPY_FILENAME="pypy${PYTHON_VERSION}-v${PYPY_VERSION}-${PYPY_ARCH}" +PYPI_URL="https://downloads.python.org/pypy/${PYPY_FILENAME}.tar.bz2" + +mkdir -p $BINDIR + +cd $BINDIR + +TAR_FILE=pyp.tar.bz2 +wget -O "${TAR_FILE}" "${PYPI_URL}" +echo "${PYPI_HASH} ${TAR_FILE}" | sha256sum -c - +tar -xjf "${TAR_FILE}" && rm "${TAR_FILE}" +mv -n "${PYPY_FILENAME}" pypy3 + +ln -s ./pypy3/bin/pypy3 python +$BINDIR/python --version + +# install PyYAML +./python -m ensurepip +./python -m pip install pyyaml + +touch $BINDIR/.bootstrapped diff --git a/kubespray/roles/bootstrap-os/handlers/main.yml b/kubespray/roles/bootstrap-os/handlers/main.yml new file mode 100644 index 0000000..7c8c4fe --- /dev/null +++ b/kubespray/roles/bootstrap-os/handlers/main.yml @@ -0,0 +1,4 @@ +--- +- name: RHEL auto-attach subscription + command: /sbin/subscription-manager attach --auto + become: true diff --git a/kubespray/roles/bootstrap-os/molecule/default/converge.yml b/kubespray/roles/bootstrap-os/molecule/default/converge.yml new file mode 100644 index 0000000..1f44ec9 --- /dev/null +++ b/kubespray/roles/bootstrap-os/molecule/default/converge.yml @@ -0,0 +1,6 @@ +--- +- name: Converge + hosts: all + gather_facts: no + roles: + - role: bootstrap-os diff --git a/kubespray/roles/bootstrap-os/molecule/default/molecule.yml b/kubespray/roles/bootstrap-os/molecule/default/molecule.yml new file mode 100644 index 0000000..76e5b7a --- /dev/null +++ b/kubespray/roles/bootstrap-os/molecule/default/molecule.yml @@ -0,0 +1,53 @@ +--- +role_name_check: 1 +dependency: + name: galaxy +driver: + name: vagrant + provider: + name: libvirt +platforms: + - name: ubuntu20 + box: generic/ubuntu2004 + cpus: 1 + memory: 512 + provider_options: + driver: kvm + - name: ubuntu22 + box: generic/ubuntu2204 + cpus: 1 + memory: 1024 + provider_options: + driver: kvm + - name: centos7 + box: centos/7 + cpus: 1 + memory: 512 + provider_options: + driver: kvm + - name: almalinux8 + box: almalinux/8 + cpus: 1 + memory: 512 + provider_options: + driver: kvm + - name: debian10 + box: generic/debian10 + cpus: 1 + memory: 512 + provider_options: + driver: kvm +provisioner: + name: ansible + config_options: + defaults: + callbacks_enabled: profile_tasks + timeout: 120 + inventory: + group_vars: + all: + user: + name: foo + comment: My test comment +verifier: + name: testinfra diff --git a/kubespray/roles/bootstrap-os/molecule/default/tests/test_default.py b/kubespray/roles/bootstrap-os/molecule/default/tests/test_default.py new file mode 100644 index 0000000..64c59dd --- /dev/null +++ b/kubespray/roles/bootstrap-os/molecule/default/tests/test_default.py @@ -0,0 +1,11 @@ +import os + +import testinfra.utils.ansible_runner + +testinfra_hosts = testinfra.utils.ansible_runner.AnsibleRunner( + os.environ['MOLECULE_INVENTORY_FILE'] +).get_hosts('all') + + +def test_python(host): + assert host.exists('python3') or host.exists('python') diff --git a/kubespray/roles/bootstrap-os/tasks/bootstrap-amazon.yml b/kubespray/roles/bootstrap-os/tasks/bootstrap-amazon.yml new file mode 100644 index 0000000..2b4d665 --- /dev/null +++ b/kubespray/roles/bootstrap-os/tasks/bootstrap-amazon.yml @@ -0,0 +1,13 @@ +--- +- name: Enable EPEL repo for Amazon Linux + yum_repository: + name: epel + file: epel + description: Extra Packages for Enterprise Linux 7 - $basearch + baseurl: http://download.fedoraproject.org/pub/epel/7/$basearch + gpgcheck: yes + gpgkey: http://download.fedoraproject.org/pub/epel/RPM-GPG-KEY-EPEL-7 + skip_if_unavailable: yes + enabled: yes + repo_gpgcheck: no + when: epel_enabled diff --git a/kubespray/roles/bootstrap-os/tasks/bootstrap-centos.yml b/kubespray/roles/bootstrap-os/tasks/bootstrap-centos.yml new file mode 100644 index 0000000..5d543ae --- /dev/null +++ b/kubespray/roles/bootstrap-os/tasks/bootstrap-centos.yml @@ -0,0 +1,118 @@ +--- +- name: Gather host facts to get ansible_distribution_version ansible_distribution_major_version + setup: + gather_subset: '!all' + filter: ansible_distribution_*version + +- name: Add proxy to yum.conf or dnf.conf if http_proxy is defined + community.general.ini_file: + path: "{{ ((ansible_distribution_major_version | int) < 8) | ternary('/etc/yum.conf', '/etc/dnf/dnf.conf') }}" + section: main + option: proxy + value: "{{ http_proxy | default(omit) }}" + state: "{{ http_proxy | default(False) | ternary('present', 'absent') }}" + no_extra_spaces: true + mode: 0644 + become: true + when: not skip_http_proxy_on_os_packages + +# For Oracle Linux install public repo +- name: Download Oracle Linux public yum repo + get_url: + url: https://yum.oracle.com/public-yum-ol7.repo + dest: /etc/yum.repos.d/public-yum-ol7.repo + mode: 0644 + when: + - use_oracle_public_repo | default(true) + - '''ID="ol"'' in os_release.stdout_lines' + - (ansible_distribution_version | float) < 7.6 + environment: "{{ proxy_env }}" + +- name: Enable Oracle Linux repo + community.general.ini_file: + dest: /etc/yum.repos.d/public-yum-ol7.repo + section: "{{ item }}" + option: enabled + value: "1" + mode: 0644 + with_items: + - ol7_latest + - ol7_addons + - ol7_developer_EPEL + when: + - use_oracle_public_repo | default(true) + - '''ID="ol"'' in os_release.stdout_lines' + - (ansible_distribution_version | float) < 7.6 + +- name: Install EPEL for Oracle Linux repo package + package: + name: "oracle-epel-release-el{{ ansible_distribution_major_version }}" + state: present + when: + - use_oracle_public_repo | default(true) + - '''ID="ol"'' in os_release.stdout_lines' + - (ansible_distribution_version | float) >= 7.6 + +- name: Enable Oracle Linux repo + community.general.ini_file: + dest: "/etc/yum.repos.d/oracle-linux-ol{{ ansible_distribution_major_version }}.repo" + section: "ol{{ ansible_distribution_major_version }}_addons" + option: "{{ item.option }}" + value: "{{ item.value }}" + mode: 0644 + with_items: + - { option: "name", value: "ol{{ ansible_distribution_major_version }}_addons" } + - { option: "enabled", value: "1" } + - { option: "baseurl", value: "http://yum.oracle.com/repo/OracleLinux/OL{{ ansible_distribution_major_version }}/addons/$basearch/" } + when: + - use_oracle_public_repo | default(true) + - '''ID="ol"'' in os_release.stdout_lines' + - (ansible_distribution_version | float) >= 7.6 + +- name: Enable Centos extra repo for Oracle Linux + community.general.ini_file: + dest: "/etc/yum.repos.d/centos-extras.repo" + section: "extras" + option: "{{ item.option }}" + value: "{{ item.value }}" + mode: 0644 + with_items: + - { option: "name", value: "CentOS-{{ ansible_distribution_major_version }} - Extras" } + - { option: "enabled", value: "1" } + - { option: "gpgcheck", value: "0" } + - { option: "baseurl", value: "http://mirror.centos.org/{{ 'altarch' if (ansible_distribution_major_version | int) <= 7 and ansible_architecture == 'aarch64' else 'centos' }}/{{ ansible_distribution_major_version }}/extras/$basearch/{% if ansible_distribution_major_version | int > 7 %}os/{% endif %}" } + when: + - use_oracle_public_repo | default(true) + - '''ID="ol"'' in os_release.stdout_lines' + - (ansible_distribution_version | float) >= 7.6 + - (ansible_distribution_version | float) < 9 + +# CentOS ships with python installed + +- name: Check presence of fastestmirror.conf + stat: + path: /etc/yum/pluginconf.d/fastestmirror.conf + get_attributes: no + get_checksum: no + get_mime: no + register: fastestmirror + +# the fastestmirror plugin can actually slow down Ansible deployments +- name: Disable fastestmirror plugin if requested + lineinfile: + dest: /etc/yum/pluginconf.d/fastestmirror.conf + regexp: "^enabled=.*" + line: "enabled=0" + state: present + become: true + when: + - fastestmirror.stat.exists + - not centos_fastestmirror_enabled + +# libselinux-python is required on SELinux enabled hosts +# See https://docs.ansible.com/ansible/latest/installation_guide/intro_installation.html#managed-node-requirements +- name: Install libselinux python package + package: + name: "{{ ((ansible_distribution_major_version | int) < 8) | ternary('libselinux-python', 'python3-libselinux') }}" + state: present + become: true diff --git a/kubespray/roles/bootstrap-os/tasks/bootstrap-clearlinux.yml b/kubespray/roles/bootstrap-os/tasks/bootstrap-clearlinux.yml new file mode 100644 index 0000000..de42e3c --- /dev/null +++ b/kubespray/roles/bootstrap-os/tasks/bootstrap-clearlinux.yml @@ -0,0 +1,16 @@ +--- +# ClearLinux ships with Python installed + +- name: Install basic package to run containers + package: + name: containers-basic + state: present + +- name: Make sure docker service is enabled + systemd: + name: docker + masked: false + enabled: true + daemon_reload: true + state: started + become: true diff --git a/kubespray/roles/bootstrap-os/tasks/bootstrap-coreos.yml b/kubespray/roles/bootstrap-os/tasks/bootstrap-coreos.yml new file mode 100644 index 0000000..737a7ec --- /dev/null +++ b/kubespray/roles/bootstrap-os/tasks/bootstrap-coreos.yml @@ -0,0 +1,37 @@ +--- +# CoreOS ships without Python installed + +- name: Check if bootstrap is needed + raw: stat /opt/bin/.bootstrapped + register: need_bootstrap + failed_when: false + changed_when: false + tags: + - facts + +- name: Force binaries directory for Container Linux by CoreOS and Flatcar + set_fact: + bin_dir: "/opt/bin" + tags: + - facts + +- name: Run bootstrap.sh + script: bootstrap.sh + become: true + environment: "{{ proxy_env }}" + when: + - need_bootstrap.rc != 0 + +- name: Set the ansible_python_interpreter fact + set_fact: + ansible_python_interpreter: "{{ bin_dir }}/python" + tags: + - facts + +- name: Disable auto-upgrade + systemd: + name: locksmithd.service + masked: true + state: stopped + when: + - coreos_locksmithd_disable diff --git a/kubespray/roles/bootstrap-os/tasks/bootstrap-debian.yml b/kubespray/roles/bootstrap-os/tasks/bootstrap-debian.yml new file mode 100644 index 0000000..47bad20 --- /dev/null +++ b/kubespray/roles/bootstrap-os/tasks/bootstrap-debian.yml @@ -0,0 +1,76 @@ +--- +# Some Debian based distros ship without Python installed + +- name: Check if bootstrap is needed + raw: which python3 + register: need_bootstrap + failed_when: false + changed_when: false + # This command should always run, even in check mode + check_mode: false + tags: + - facts + +- name: Check http::proxy in apt configuration files + raw: apt-config dump | grep -qsi 'Acquire::http::proxy' + register: need_http_proxy + failed_when: false + changed_when: false + # This command should always run, even in check mode + check_mode: false + +- name: Add http_proxy to /etc/apt/apt.conf if http_proxy is defined + raw: echo 'Acquire::http::proxy "{{ http_proxy }}";' >> /etc/apt/apt.conf + become: true + when: + - http_proxy is defined + - need_http_proxy.rc != 0 + - not skip_http_proxy_on_os_packages + +- name: Check https::proxy in apt configuration files + raw: apt-config dump | grep -qsi 'Acquire::https::proxy' + register: need_https_proxy + failed_when: false + changed_when: false + # This command should always run, even in check mode + check_mode: false + +- name: Add https_proxy to /etc/apt/apt.conf if https_proxy is defined + raw: echo 'Acquire::https::proxy "{{ https_proxy }}";' >> /etc/apt/apt.conf + become: true + when: + - https_proxy is defined + - need_https_proxy.rc != 0 + - not skip_http_proxy_on_os_packages + +- name: Install python3 + raw: + apt-get update && \ + DEBIAN_FRONTEND=noninteractive apt-get install -y python3-minimal + become: true + when: + - need_bootstrap.rc != 0 + +- name: Update Apt cache + raw: apt-get update --allow-releaseinfo-change + become: true + when: + - '''ID=debian'' in os_release.stdout_lines' + - '''VERSION_ID="10"'' in os_release.stdout_lines or ''VERSION_ID="11"'' in os_release.stdout_lines' + register: bootstrap_update_apt_result + changed_when: + - '"changed its" in bootstrap_update_apt_result.stdout' + - '"value from" in bootstrap_update_apt_result.stdout' + ignore_errors: true + +- name: Set the ansible_python_interpreter fact + set_fact: + ansible_python_interpreter: "/usr/bin/python3" + +# Workaround for https://github.com/ansible/ansible/issues/25543 +- name: Install dbus for the hostname module + package: + name: dbus + state: present + use: apt + become: true diff --git a/kubespray/roles/bootstrap-os/tasks/bootstrap-fedora-coreos.yml b/kubespray/roles/bootstrap-os/tasks/bootstrap-fedora-coreos.yml new file mode 100644 index 0000000..91dc020 --- /dev/null +++ b/kubespray/roles/bootstrap-os/tasks/bootstrap-fedora-coreos.yml @@ -0,0 +1,46 @@ +--- + +- name: Check if bootstrap is needed + raw: which python + register: need_bootstrap + failed_when: false + changed_when: false + tags: + - facts + +- name: Remove podman network cni + raw: "podman network rm podman" + become: true + ignore_errors: true # noqa ignore-errors + when: need_bootstrap.rc != 0 + +- name: Clean up possible pending packages on fedora coreos + raw: "export http_proxy={{ http_proxy | default('') }};rpm-ostree cleanup -p }}" + become: true + when: need_bootstrap.rc != 0 + +- name: Install required packages on fedora coreos + raw: "export http_proxy={{ http_proxy | default('') }};rpm-ostree install --allow-inactive {{ fedora_coreos_packages | join(' ') }}" + become: true + when: need_bootstrap.rc != 0 + +- name: Reboot immediately for updated ostree + raw: "nohup bash -c 'sleep 5s && shutdown -r now'" + become: true + ignore_errors: true # noqa ignore-errors + ignore_unreachable: yes + when: need_bootstrap.rc != 0 + +- name: Wait for the reboot to complete + wait_for_connection: + timeout: 240 + connect_timeout: 20 + delay: 5 + sleep: 5 + when: need_bootstrap.rc != 0 + +- name: Store the fact if this is an fedora core os host + set_fact: + is_fedora_coreos: True + tags: + - facts diff --git a/kubespray/roles/bootstrap-os/tasks/bootstrap-fedora.yml b/kubespray/roles/bootstrap-os/tasks/bootstrap-fedora.yml new file mode 100644 index 0000000..4ce77b4 --- /dev/null +++ b/kubespray/roles/bootstrap-os/tasks/bootstrap-fedora.yml @@ -0,0 +1,36 @@ +--- +# Some Fedora based distros ship without Python installed + +- name: Check if bootstrap is needed + raw: which python + register: need_bootstrap + failed_when: false + changed_when: false + tags: + - facts + +- name: Add proxy to dnf.conf if http_proxy is defined + community.general.ini_file: + path: "/etc/dnf/dnf.conf" + section: main + option: proxy + value: "{{ http_proxy | default(omit) }}" + state: "{{ http_proxy | default(False) | ternary('present', 'absent') }}" + no_extra_spaces: true + mode: 0644 + become: true + when: not skip_http_proxy_on_os_packages + +- name: Install python3 on fedora + raw: "dnf install --assumeyes --quiet python3" + become: true + when: + - need_bootstrap.rc != 0 + +# libselinux-python3 is required on SELinux enabled hosts +# See https://docs.ansible.com/ansible/latest/installation_guide/intro_installation.html#managed-node-requirements +- name: Install libselinux-python3 + package: + name: libselinux-python3 + state: present + become: true diff --git a/kubespray/roles/bootstrap-os/tasks/bootstrap-flatcar.yml b/kubespray/roles/bootstrap-os/tasks/bootstrap-flatcar.yml new file mode 100644 index 0000000..b0f3a9e --- /dev/null +++ b/kubespray/roles/bootstrap-os/tasks/bootstrap-flatcar.yml @@ -0,0 +1,37 @@ +--- +# Flatcar Container Linux ships without Python installed + +- name: Check if bootstrap is needed + raw: stat /opt/bin/.bootstrapped + register: need_bootstrap + failed_when: false + changed_when: false + tags: + - facts + +- name: Force binaries directory for Flatcar Container Linux by Kinvolk + set_fact: + bin_dir: "/opt/bin" + tags: + - facts + +- name: Run bootstrap.sh + script: bootstrap.sh + become: true + environment: "{{ proxy_env }}" + when: + - need_bootstrap.rc != 0 + +- name: Set the ansible_python_interpreter fact + set_fact: + ansible_python_interpreter: "{{ bin_dir }}/python" + tags: + - facts + +- name: Disable auto-upgrade + systemd: + name: locksmithd.service + masked: true + state: stopped + when: + - coreos_locksmithd_disable diff --git a/kubespray/roles/bootstrap-os/tasks/bootstrap-opensuse.yml b/kubespray/roles/bootstrap-os/tasks/bootstrap-opensuse.yml new file mode 100644 index 0000000..9b69dcd --- /dev/null +++ b/kubespray/roles/bootstrap-os/tasks/bootstrap-opensuse.yml @@ -0,0 +1,85 @@ +--- +# OpenSUSE ships with Python installed +- name: Gather host facts to get ansible_distribution_version ansible_distribution_major_version + setup: + gather_subset: '!all' + filter: ansible_distribution_*version + +- name: Check that /etc/sysconfig/proxy file exists + stat: + path: /etc/sysconfig/proxy + get_attributes: no + get_checksum: no + get_mime: no + register: stat_result + +- name: Create the /etc/sysconfig/proxy empty file + file: # noqa risky-file-permissions + path: /etc/sysconfig/proxy + state: touch + when: + - http_proxy is defined or https_proxy is defined + - not stat_result.stat.exists + +- name: Set the http_proxy in /etc/sysconfig/proxy + lineinfile: + path: /etc/sysconfig/proxy + regexp: '^HTTP_PROXY=' + line: 'HTTP_PROXY="{{ http_proxy }}"' + become: true + when: + - http_proxy is defined + +- name: Set the https_proxy in /etc/sysconfig/proxy + lineinfile: + path: /etc/sysconfig/proxy + regexp: '^HTTPS_PROXY=' + line: 'HTTPS_PROXY="{{ https_proxy }}"' + become: true + when: + - https_proxy is defined + +- name: Enable proxies + lineinfile: + path: /etc/sysconfig/proxy + regexp: '^PROXY_ENABLED=' + line: 'PROXY_ENABLED="yes"' + become: true + when: + - http_proxy is defined or https_proxy is defined + +# Required for zypper module +- name: Install python-xml + shell: zypper refresh && zypper --non-interactive install python-xml + changed_when: false + become: true + tags: + - facts + +# Without this package, the get_url module fails when trying to handle https +- name: Install python-cryptography + community.general.zypper: + name: python-cryptography + state: present + update_cache: true + become: true + when: + - ansible_distribution_version is version('15.4', '<') + +- name: Install python3-cryptography + community.general.zypper: + name: python3-cryptography + state: present + update_cache: true + become: true + when: + - ansible_distribution_version is version('15.4', '>=') + +# Nerdctl needs some basic packages to get an environment up +- name: Install basic dependencies + community.general.zypper: + name: + - iptables + - apparmor-parser + state: present + become: true diff --git a/kubespray/roles/bootstrap-os/tasks/bootstrap-redhat.yml b/kubespray/roles/bootstrap-os/tasks/bootstrap-redhat.yml new file mode 100644 index 0000000..c362146 --- /dev/null +++ b/kubespray/roles/bootstrap-os/tasks/bootstrap-redhat.yml @@ -0,0 +1,113 @@ +--- +- name: Gather host facts to get ansible_distribution_version ansible_distribution_major_version + setup: + gather_subset: '!all' + filter: ansible_distribution_*version + +- name: Add proxy to yum.conf or dnf.conf if http_proxy is defined + community.general.ini_file: + path: "{{ ((ansible_distribution_major_version | int) < 8) | ternary('/etc/yum.conf', '/etc/dnf/dnf.conf') }}" + section: main + option: proxy + value: "{{ http_proxy | default(omit) }}" + state: "{{ http_proxy | default(False) | ternary('present', 'absent') }}" + no_extra_spaces: true + mode: 0644 + become: true + when: not skip_http_proxy_on_os_packages + +- name: Add proxy to RHEL subscription-manager if http_proxy is defined + command: /sbin/subscription-manager config --server.proxy_hostname={{ http_proxy | regex_replace(':\d+$') | regex_replace('^.*://') }} --server.proxy_port={{ http_proxy | regex_replace('^.*:') }} + become: true + when: + - not skip_http_proxy_on_os_packages + - http_proxy is defined + +- name: Check RHEL subscription-manager status + command: /sbin/subscription-manager status + register: rh_subscription_status + changed_when: "rh_subscription_status.rc != 0" + ignore_errors: true # noqa ignore-errors + become: true + +- name: RHEL subscription Organization ID/Activation Key registration + community.general.redhat_subscription: + state: present + org_id: "{{ rh_subscription_org_id }}" + activationkey: "{{ rh_subscription_activation_key }}" + force_register: true + notify: RHEL auto-attach subscription + become: true + when: + - rh_subscription_org_id is defined + - rh_subscription_status.changed + +# this task has no_log set to prevent logging security sensitive information such as subscription passwords +- name: RHEL subscription Username/Password registration + community.general.redhat_subscription: + state: present + username: "{{ rh_subscription_username }}" + password: "{{ rh_subscription_password }}" + auto_attach: true + force_register: true + syspurpose: + usage: "{{ rh_subscription_usage }}" + role: "{{ rh_subscription_role }}" + service_level_agreement: "{{ rh_subscription_sla }}" + sync: true + notify: RHEL auto-attach subscription + become: true + no_log: "{{ not (unsafe_show_logs | bool) }}" + when: + - rh_subscription_username is defined + - rh_subscription_status.changed + +# container-selinux is in extras repo +- name: Enable RHEL 7 repos + community.general.rhsm_repository: + name: + - "rhel-7-server-rpms" + - "rhel-7-server-extras-rpms" + state: "{{ 'enabled' if (rhel_enable_repos | default(True) | bool) else 'disabled' }}" + when: + - ansible_distribution_major_version == "7" + - (not rh_subscription_status.changed) or (rh_subscription_username is defined) or (rh_subscription_org_id is defined) + +# container-selinux is in appstream repo +- name: Enable RHEL 8 repos + community.general.rhsm_repository: + name: + - "rhel-8-for-*-baseos-rpms" + - "rhel-8-for-*-appstream-rpms" + state: "{{ 'enabled' if (rhel_enable_repos | default(True) | bool) else 'disabled' }}" + when: + - ansible_distribution_major_version == "8" + - (not rh_subscription_status.changed) or (rh_subscription_username is defined) or (rh_subscription_org_id is defined) + +- name: Check presence of fastestmirror.conf + stat: + path: /etc/yum/pluginconf.d/fastestmirror.conf + get_attributes: no + get_checksum: no + get_mime: no + register: fastestmirror + +# the fastestmirror plugin can actually slow down Ansible deployments +- name: Disable fastestmirror plugin if requested + lineinfile: + dest: /etc/yum/pluginconf.d/fastestmirror.conf + regexp: "^enabled=.*" + line: "enabled=0" + state: present + become: true + when: + - fastestmirror.stat.exists + - not centos_fastestmirror_enabled + +# libselinux-python is required on SELinux enabled hosts +# See https://docs.ansible.com/ansible/latest/installation_guide/intro_installation.html#managed-node-requirements +- name: Install libselinux python package + package: + name: "{{ ((ansible_distribution_major_version | int) < 8) | ternary('libselinux-python', 'python3-libselinux') }}" + state: present + become: true diff --git a/kubespray/roles/bootstrap-os/tasks/main.yml b/kubespray/roles/bootstrap-os/tasks/main.yml new file mode 100644 index 0000000..73c9e06 --- /dev/null +++ b/kubespray/roles/bootstrap-os/tasks/main.yml @@ -0,0 +1,109 @@ +--- +- name: Fetch /etc/os-release + raw: cat /etc/os-release + register: os_release + changed_when: false + # This command should always run, even in check mode + check_mode: false + +- name: Bootstrap CentOS + include_tasks: bootstrap-centos.yml + when: '''ID="centos"'' in os_release.stdout_lines or ''ID="ol"'' in os_release.stdout_lines or ''ID="almalinux"'' in os_release.stdout_lines or ''ID="rocky"'' in os_release.stdout_lines or ''ID="kylin"'' in os_release.stdout_lines or ''ID="uos"'' in os_release.stdout_lines or ''ID="openEuler"'' in os_release.stdout_lines' + +- name: Bootstrap Amazon + include_tasks: bootstrap-amazon.yml + when: '''ID="amzn"'' in os_release.stdout_lines' + +- name: Bootstrap RedHat + include_tasks: bootstrap-redhat.yml + when: '''ID="rhel"'' in os_release.stdout_lines' + +- name: Bootstrap Clear Linux + include_tasks: bootstrap-clearlinux.yml + when: '''ID=clear-linux-os'' in os_release.stdout_lines' + +# Fedora CoreOS +- name: Bootstrap Fedora CoreOS + include_tasks: bootstrap-fedora-coreos.yml + when: + - '''ID=fedora'' in os_release.stdout_lines' + - '''VARIANT_ID=coreos'' in os_release.stdout_lines' + +- name: Bootstrap Flatcar + include_tasks: bootstrap-flatcar.yml + when: '''ID=flatcar'' in os_release.stdout_lines' + +- name: Bootstrap Debian + include_tasks: bootstrap-debian.yml + when: '''ID=debian'' in os_release.stdout_lines or ''ID=ubuntu'' in os_release.stdout_lines' + +# Fedora "classic" +- name: Boostrap Fedora + include_tasks: bootstrap-fedora.yml + when: + - '''ID=fedora'' in os_release.stdout_lines' + - '''VARIANT_ID=coreos'' not in os_release.stdout_lines' + +- name: Bootstrap OpenSUSE + include_tasks: bootstrap-opensuse.yml + when: '''ID="opensuse-leap"'' in os_release.stdout_lines or ''ID="opensuse-tumbleweed"'' in os_release.stdout_lines' + +- name: Create remote_tmp for it is used by another module + file: + path: "{{ ansible_remote_tmp | default('~/.ansible/tmp') }}" + state: directory + mode: 0700 + +# Workaround for https://github.com/ansible/ansible/issues/42726 +# (1/3) +- name: Gather host facts to get ansible_os_family + setup: + gather_subset: '!all' + filter: ansible_* + +- name: Assign inventory name to unconfigured hostnames (non-CoreOS, non-Flatcar, Suse and ClearLinux, non-Fedora) + hostname: + name: "{{ inventory_hostname }}" + when: + - override_system_hostname + - ansible_os_family not in ['Suse', 'Flatcar', 'Flatcar Container Linux by Kinvolk', 'ClearLinux'] + - not ansible_distribution == "Fedora" + - not is_fedora_coreos + +# (2/3) +- name: Assign inventory name to unconfigured hostnames (CoreOS, Flatcar, Suse, ClearLinux and Fedora only) + command: "hostnamectl set-hostname {{ inventory_hostname }}" + register: hostname_changed + become: true + changed_when: false + when: > + override_system_hostname + and (ansible_os_family in ['Suse', 'Flatcar', 'Flatcar Container Linux by Kinvolk', 'ClearLinux'] + or is_fedora_coreos + or ansible_distribution == "Fedora") + +# (3/3) +- name: Update hostname fact (CoreOS, Flatcar, Suse, ClearLinux and Fedora only) + setup: + gather_subset: '!all' + filter: ansible_hostname + when: > + override_system_hostname + and (ansible_os_family in ['Suse', 'Flatcar', 'Flatcar Container Linux by Kinvolk', 'ClearLinux'] + or is_fedora_coreos + or ansible_distribution == "Fedora") + +- name: Install ceph-commmon package + package: + name: + - ceph-common + state: present + when: rbd_provisioner_enabled | default(false) + +- name: Ensure bash_completion.d folder exists + file: + name: /etc/bash_completion.d/ + state: directory + owner: root + group: root + mode: 0755 diff --git a/kubespray/roles/container-engine/containerd-common/defaults/main.yml b/kubespray/roles/container-engine/containerd-common/defaults/main.yml new file mode 100644 index 0000000..ae1c6e0 --- /dev/null +++ b/kubespray/roles/container-engine/containerd-common/defaults/main.yml @@ -0,0 +1,17 @@ +--- +# We keep these variables around to allow migration from package +# manager controlled installs to direct download ones. +containerd_package: 'containerd.io' +yum_repo_dir: /etc/yum.repos.d + +# Keep minimal repo information around for cleanup +containerd_repo_info: + repos: + +# Ubuntu docker-ce repo +containerd_ubuntu_repo_base_url: "https://download.docker.com/linux/ubuntu" +containerd_ubuntu_repo_component: "stable" + +# Debian docker-ce repo +containerd_debian_repo_base_url: "https://download.docker.com/linux/debian" +containerd_debian_repo_component: "stable" diff --git a/kubespray/roles/container-engine/containerd-common/meta/main.yml b/kubespray/roles/container-engine/containerd-common/meta/main.yml new file mode 100644 index 0000000..61d3ffe --- /dev/null +++ b/kubespray/roles/container-engine/containerd-common/meta/main.yml @@ -0,0 +1,2 @@ +--- +allow_duplicates: true diff --git a/kubespray/roles/container-engine/containerd-common/tasks/main.yml b/kubespray/roles/container-engine/containerd-common/tasks/main.yml new file mode 100644 index 0000000..d0cf1f1 --- /dev/null +++ b/kubespray/roles/container-engine/containerd-common/tasks/main.yml @@ -0,0 +1,31 @@ +--- +- name: Containerd-common | check if fedora coreos + stat: + path: /run/ostree-booted + get_attributes: no + get_checksum: no + get_mime: no + register: ostree + +- name: Containerd-common | set is_ostree + set_fact: + is_ostree: "{{ ostree.stat.exists }}" + +- name: Containerd-common | gather os specific variables + include_vars: "{{ item }}" + with_first_found: + - files: + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_release | lower }}-{{ host_architecture }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_release | lower }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_major_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}-{{ host_architecture }}.yml" + - "{{ ansible_distribution | lower }}.yml" + - "{{ ansible_os_family | lower }}-{{ host_architecture }}.yml" + - "{{ ansible_os_family | lower }}.yml" + - defaults.yml + paths: + - ../vars + skip: true + tags: + - facts diff --git a/kubespray/roles/container-engine/containerd-common/vars/amazon.yml b/kubespray/roles/container-engine/containerd-common/vars/amazon.yml new file mode 100644 index 0000000..0568169 --- /dev/null +++ b/kubespray/roles/container-engine/containerd-common/vars/amazon.yml @@ -0,0 +1,2 @@ +--- +containerd_package: containerd diff --git a/kubespray/roles/container-engine/containerd-common/vars/suse.yml b/kubespray/roles/container-engine/containerd-common/vars/suse.yml new file mode 100644 index 0000000..0568169 --- /dev/null +++ b/kubespray/roles/container-engine/containerd-common/vars/suse.yml @@ -0,0 +1,2 @@ +--- +containerd_package: containerd diff --git a/kubespray/roles/container-engine/containerd/defaults/main.yml b/kubespray/roles/container-engine/containerd/defaults/main.yml new file mode 100644 index 0000000..05cfd95 --- /dev/null +++ b/kubespray/roles/container-engine/containerd/defaults/main.yml @@ -0,0 +1,106 @@ +--- +containerd_storage_dir: "/var/lib/containerd" +containerd_state_dir: "/run/containerd" +containerd_systemd_dir: "/etc/systemd/system/containerd.service.d" +# The default value is not -999 here because containerd's oom_score_adj has been +# set to the -999 even if containerd_oom_score is 0. +# Ref: https://github.com/kubernetes-sigs/kubespray/pull/9275#issuecomment-1246499242 +containerd_oom_score: 0 + +# containerd_default_runtime: "runc" +# containerd_snapshotter: "native" + +containerd_runc_runtime: + name: runc + type: "io.containerd.runc.v2" + engine: "" + root: "" + base_runtime_spec: cri-base.json + options: + systemdCgroup: "{{ containerd_use_systemd_cgroup | ternary('true', 'false') }}" + +containerd_additional_runtimes: [] +# Example for Kata Containers as additional runtime: +# - name: kata +# type: "io.containerd.kata.v2" +# engine: "" +# root: "" + +containerd_base_runtime_spec_rlimit_nofile: 65535 + +containerd_default_base_runtime_spec_patch: + process: + rlimits: + - type: RLIMIT_NOFILE + hard: "{{ containerd_base_runtime_spec_rlimit_nofile }}" + soft: "{{ containerd_base_runtime_spec_rlimit_nofile }}" + +containerd_base_runtime_specs: + cri-base.json: "{{ containerd_default_base_runtime_spec | combine(containerd_default_base_runtime_spec_patch, recursive=1) }}" + +containerd_grpc_max_recv_message_size: 16777216 +containerd_grpc_max_send_message_size: 16777216 + +containerd_debug_level: "info" + +containerd_metrics_address: "" + +containerd_metrics_grpc_histogram: false + +containerd_registries: + "docker.io": "https://registry-1.docker.io" + +containerd_registries_mirrors: + - prefix: docker.io + mirrors: + - host: https://registry-1.docker.io + capabilities: ["pull", "resolve"] + skip_verify: false + +containerd_max_container_log_line_size: -1 + +# If enabled it will allow non root users to use port numbers <1024 +containerd_enable_unprivileged_ports: false +# If enabled it will allow non root users to use icmp sockets +containerd_enable_unprivileged_icmp: false + +containerd_cfg_dir: /etc/containerd + +# Extra config to be put in {{ containerd_cfg_dir }}/config.toml literally +containerd_extra_args: '' + +# Configure registry auth (if applicable to secure/insecure registries) +containerd_registry_auth: [] +# - registry: 10.0.0.2:5000 +# username: user +# password: pass + +# Configure containerd service +containerd_limit_proc_num: "infinity" +containerd_limit_core: "infinity" +containerd_limit_open_file_num: "infinity" +containerd_limit_mem_lock: "infinity" + +# If enabled it will use config_path and config to be put in {{ containerd_cfg_dir }}/certs.d/ +containerd_use_config_path: false + +# OS distributions that already support containerd +containerd_supported_distributions: + - "CentOS" + - "OracleLinux" + - "RedHat" + - "Ubuntu" + - "Debian" + - "Fedora" + - "AlmaLinux" + - "Rocky" + - "Amazon" + - "Flatcar" + - "Flatcar Container Linux by Kinvolk" + - "Suse" + - "openSUSE Leap" + - "openSUSE Tumbleweed" + - "Kylin Linux Advanced Server" + - "UnionTech" + - "UniontechOS" + - "openEuler" diff --git a/kubespray/roles/container-engine/containerd/handlers/main.yml b/kubespray/roles/container-engine/containerd/handlers/main.yml new file mode 100644 index 0000000..3c132bd --- /dev/null +++ b/kubespray/roles/container-engine/containerd/handlers/main.yml @@ -0,0 +1,21 @@ +--- +- name: Restart containerd + command: /bin/true + notify: + - Containerd | restart containerd + - Containerd | wait for containerd + +- name: Containerd | restart containerd + systemd: + name: containerd + state: restarted + enabled: yes + daemon-reload: yes + masked: no + +- name: Containerd | wait for containerd + command: "{{ containerd_bin_dir }}/ctr images ls -q" + register: containerd_ready + retries: 8 + delay: 4 + until: containerd_ready.rc == 0 diff --git a/kubespray/roles/container-engine/containerd/handlers/reset.yml b/kubespray/roles/container-engine/containerd/handlers/reset.yml new file mode 100644 index 0000000..ed97d53 --- /dev/null +++ b/kubespray/roles/container-engine/containerd/handlers/reset.yml @@ -0,0 +1 @@ +--- diff --git a/kubespray/roles/container-engine/containerd/meta/main.yml b/kubespray/roles/container-engine/containerd/meta/main.yml new file mode 100644 index 0000000..41c5b6a --- /dev/null +++ b/kubespray/roles/container-engine/containerd/meta/main.yml @@ -0,0 +1,6 @@ +--- +dependencies: + - role: container-engine/containerd-common + - role: container-engine/runc + - role: container-engine/crictl + - role: container-engine/nerdctl diff --git a/kubespray/roles/container-engine/containerd/molecule/default/converge.yml b/kubespray/roles/container-engine/containerd/molecule/default/converge.yml new file mode 100644 index 0000000..7847871 --- /dev/null +++ b/kubespray/roles/container-engine/containerd/molecule/default/converge.yml @@ -0,0 +1,9 @@ +--- +- name: Converge + hosts: all + become: true + vars: + container_manager: containerd + roles: + - role: kubespray-defaults + - role: container-engine/containerd diff --git a/kubespray/roles/container-engine/containerd/molecule/default/molecule.yml b/kubespray/roles/container-engine/containerd/molecule/default/molecule.yml new file mode 100644 index 0000000..4c5c48f --- /dev/null +++ b/kubespray/roles/container-engine/containerd/molecule/default/molecule.yml @@ -0,0 +1,47 @@ +--- +role_name_check: 1 +driver: + name: vagrant + provider: + name: libvirt +platforms: + - name: ubuntu20 + box: generic/ubuntu2004 + cpus: 1 + memory: 1024 + groups: + - kube_control_plane + - kube_node + - k8s_cluster + provider_options: + driver: kvm + - name: debian11 + box: generic/debian11 + cpus: 1 + memory: 1024 + groups: + - kube_control_plane + - kube_node + - k8s_cluster + provider_options: + driver: kvm + - name: almalinux8 + box: almalinux/8 + cpus: 1 + memory: 1024 + groups: + - kube_control_plane + - kube_node + - k8s_cluster + provider_options: + driver: kvm +provisioner: + name: ansible + env: + ANSIBLE_ROLES_PATH: ../../../../ + config_options: + defaults: + callbacks_enabled: profile_tasks + timeout: 120 +verifier: + name: testinfra diff --git a/kubespray/roles/container-engine/containerd/molecule/default/prepare.yml b/kubespray/roles/container-engine/containerd/molecule/default/prepare.yml new file mode 100644 index 0000000..ddc9c04 --- /dev/null +++ b/kubespray/roles/container-engine/containerd/molecule/default/prepare.yml @@ -0,0 +1,29 @@ +--- +- name: Prepare + hosts: all + gather_facts: False + become: true + vars: + ignore_assert_errors: true + roles: + - role: kubespray-defaults + - role: bootstrap-os + - role: kubernetes/preinstall + - role: adduser + user: "{{ addusers.kube }}" + tasks: + - name: Download CNI + include_tasks: "../../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.cni) }}" + +- name: Prepare CNI + hosts: all + gather_facts: False + become: true + vars: + ignore_assert_errors: true + kube_network_plugin: cni + roles: + - role: kubespray-defaults + - role: network_plugin/cni diff --git a/kubespray/roles/container-engine/containerd/molecule/default/tests/test_default.py b/kubespray/roles/container-engine/containerd/molecule/default/tests/test_default.py new file mode 100644 index 0000000..e1d9151 --- /dev/null +++ b/kubespray/roles/container-engine/containerd/molecule/default/tests/test_default.py @@ -0,0 +1,55 @@ +import os +import pytest + +import testinfra.utils.ansible_runner + +testinfra_hosts = testinfra.utils.ansible_runner.AnsibleRunner( + os.environ['MOLECULE_INVENTORY_FILE']).get_hosts('all') + + +def test_service(host): + svc = host.service("containerd") + assert svc.is_running + assert svc.is_enabled + + +def test_version(host): + crictl = "/usr/local/bin/crictl" + path = "unix:///var/run/containerd/containerd.sock" + with host.sudo(): + cmd = host.command(crictl + " --runtime-endpoint " + path + " version") + assert cmd.rc == 0 + assert "RuntimeName: containerd" in cmd.stdout + + +@pytest.mark.parametrize('image, dest', [ + ('quay.io/kubespray/hello-world:latest', '/tmp/hello-world.tar') +]) +def test_image_pull_save_load(host, image, dest): + nerdctl = "/usr/local/bin/nerdctl" + dest_file = host.file(dest) + + with host.sudo(): + pull_cmd = host.command(nerdctl + " pull " + image) + assert pull_cmd.rc ==0 + + with host.sudo(): + save_cmd = host.command(nerdctl + " save -o " + dest + " " + image) + assert save_cmd.rc == 0 + assert dest_file.exists + + with host.sudo(): + load_cmd = host.command(nerdctl + " load < " + dest) + assert load_cmd.rc == 0 + + +@pytest.mark.parametrize('image', [ + ('quay.io/kubespray/hello-world:latest') +]) +def test_run(host, image): + nerdctl = "/usr/local/bin/nerdctl" + + with host.sudo(): + cmd = host.command(nerdctl + " -n k8s.io run " + image) + assert cmd.rc == 0 + assert "Hello from Docker" in cmd.stdout diff --git a/kubespray/roles/container-engine/containerd/tasks/main.yml b/kubespray/roles/container-engine/containerd/tasks/main.yml new file mode 100644 index 0000000..43aa689 --- /dev/null +++ b/kubespray/roles/container-engine/containerd/tasks/main.yml @@ -0,0 +1,140 @@ +--- +- name: Fail containerd setup if distribution is not supported + fail: + msg: "{{ ansible_distribution }} is not supported by containerd." + when: + - not (allow_unsupported_distribution_setup | default(false)) and (ansible_distribution not in containerd_supported_distributions) + +- name: Containerd | Remove any package manager controlled containerd package + package: + name: "{{ containerd_package }}" + state: absent + when: + - not (is_ostree or (ansible_distribution == "Flatcar Container Linux by Kinvolk") or (ansible_distribution == "Flatcar")) + +- name: Containerd | Remove containerd repository + file: + path: "{{ yum_repo_dir }}/containerd.repo" + state: absent + when: + - ansible_os_family in ['RedHat'] + +- name: Containerd | Remove containerd repository + apt_repository: + repo: "{{ item }}" + state: absent + with_items: "{{ containerd_repo_info.repos }}" + when: ansible_pkg_mgr == 'apt' + +- name: Containerd | Download containerd + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.containerd) }}" + +- name: Containerd | Unpack containerd archive + unarchive: + src: "{{ downloads.containerd.dest }}" + dest: "{{ containerd_bin_dir }}" + mode: 0755 + remote_src: yes + extra_opts: + - --strip-components=1 + notify: Restart containerd + +- name: Containerd | Remove orphaned binary + file: + path: "/usr/bin/{{ item }}" + state: absent + when: + - containerd_bin_dir != "/usr/bin" + - not (is_ostree or (ansible_distribution == "Flatcar Container Linux by Kinvolk") or (ansible_distribution == "Flatcar")) + ignore_errors: true # noqa ignore-errors + with_items: + - containerd + - containerd-shim + - containerd-shim-runc-v1 + - containerd-shim-runc-v2 + - ctr + +- name: Containerd | Generate systemd service for containerd + template: + src: containerd.service.j2 + dest: /etc/systemd/system/containerd.service + mode: 0644 + notify: Restart containerd + +- name: Containerd | Ensure containerd directories exist + file: + dest: "{{ item }}" + state: directory + mode: 0755 + owner: root + group: root + with_items: + - "{{ containerd_systemd_dir }}" + - "{{ containerd_cfg_dir }}" + - "{{ containerd_storage_dir }}" + - "{{ containerd_state_dir }}" + +- name: Containerd | Write containerd proxy drop-in + template: + src: http-proxy.conf.j2 + dest: "{{ containerd_systemd_dir }}/http-proxy.conf" + mode: 0644 + notify: Restart containerd + when: http_proxy is defined or https_proxy is defined + +- name: Containerd | Generate default base_runtime_spec + register: ctr_oci_spec + command: "{{ containerd_bin_dir }}/ctr oci spec" + check_mode: false + changed_when: false + +- name: Containerd | Store generated default base_runtime_spec + set_fact: + containerd_default_base_runtime_spec: "{{ ctr_oci_spec.stdout | from_json }}" + +- name: Containerd | Write base_runtime_specs + copy: + content: "{{ item.value }}" + dest: "{{ containerd_cfg_dir }}/{{ item.key }}" + owner: "root" + mode: 0644 + with_dict: "{{ containerd_base_runtime_specs | default({}) }}" + notify: Restart containerd + +- name: Containerd | Copy containerd config file + template: + src: config.toml.j2 + dest: "{{ containerd_cfg_dir }}/config.toml" + owner: "root" + mode: 0640 + notify: Restart containerd + +- name: Containerd | Configure containerd registries + when: containerd_registries_mirrors is defined + block: + - name: Containerd | Create registry directories + file: + path: "{{ containerd_cfg_dir }}/certs.d/{{ item.prefix }}" + state: directory + mode: 0755 + loop: "{{ containerd_registries_mirrors }}" + - name: Containerd | Write hosts.toml file + template: + src: hosts.toml.j2 + dest: "{{ containerd_cfg_dir }}/certs.d/{{ item.prefix }}/hosts.toml" + mode: 0640 + loop: "{{ containerd_registries_mirrors }}" + +# you can sometimes end up in a state where everything is installed +# but containerd was not started / enabled +- name: Containerd | Flush handlers + meta: flush_handlers + +- name: Containerd | Ensure containerd is started and enabled + systemd: + name: containerd + daemon_reload: yes + enabled: yes + state: started diff --git a/kubespray/roles/container-engine/containerd/tasks/reset.yml b/kubespray/roles/container-engine/containerd/tasks/reset.yml new file mode 100644 index 0000000..517e56d --- /dev/null +++ b/kubespray/roles/container-engine/containerd/tasks/reset.yml @@ -0,0 +1,40 @@ +--- +- name: Containerd | Remove containerd repository for RedHat os family + file: + path: "{{ yum_repo_dir }}/containerd.repo" + state: absent + when: + - ansible_os_family in ['RedHat'] + tags: + - reset_containerd + +- name: Containerd | Remove containerd repository for Debian os family + apt_repository: + repo: "{{ item }}" + state: absent + with_items: "{{ containerd_repo_info.repos }}" + when: ansible_pkg_mgr == 'apt' + tags: + - reset_containerd + +- name: Containerd | Stop containerd service + service: + name: containerd + daemon_reload: true + enabled: false + state: stopped + tags: + - reset_containerd + +- name: Containerd | Remove configuration files + file: + path: "{{ item }}" + state: absent + loop: + - /etc/systemd/system/containerd.service + - "{{ containerd_systemd_dir }}" + - "{{ containerd_cfg_dir }}" + - "{{ containerd_storage_dir }}" + - "{{ containerd_state_dir }}" + tags: + - reset_containerd diff --git a/kubespray/roles/container-engine/containerd/templates/config.toml.j2 b/kubespray/roles/container-engine/containerd/templates/config.toml.j2 new file mode 100644 index 0000000..fc3ea47 --- /dev/null +++ b/kubespray/roles/container-engine/containerd/templates/config.toml.j2 @@ -0,0 +1,88 @@ +version = 2 +root = "{{ containerd_storage_dir }}" +state = "{{ containerd_state_dir }}" +oom_score = {{ containerd_oom_score }} + +[grpc] + max_recv_message_size = {{ containerd_grpc_max_recv_message_size | default(16777216) }} + max_send_message_size = {{ containerd_grpc_max_send_message_size | default(16777216) }} + +[debug] + level = "{{ containerd_debug_level | default('info') }}" + +[metrics] + address = "{{ containerd_metrics_address | default('') }}" + grpc_histogram = {{ containerd_metrics_grpc_histogram | default(false) | lower }} + +[plugins] + [plugins."io.containerd.grpc.v1.cri"] + sandbox_image = "{{ pod_infra_image_repo }}:{{ pod_infra_image_tag }}" + max_container_log_line_size = {{ containerd_max_container_log_line_size }} + enable_unprivileged_ports = {{ containerd_enable_unprivileged_ports | default(false) | lower }} + enable_unprivileged_icmp = {{ containerd_enable_unprivileged_icmp | default(false) | lower }} + [plugins."io.containerd.grpc.v1.cri".containerd] + default_runtime_name = "{{ containerd_default_runtime | default('runc') }}" + snapshotter = "{{ containerd_snapshotter | default('overlayfs') }}" + [plugins."io.containerd.grpc.v1.cri".containerd.runtimes] +{% for runtime in [containerd_runc_runtime] + containerd_additional_runtimes %} + [plugins."io.containerd.grpc.v1.cri".containerd.runtimes.{{ runtime.name }}] + runtime_type = "{{ runtime.type }}" + runtime_engine = "{{ runtime.engine }}" + runtime_root = "{{ runtime.root }}" +{% if runtime.base_runtime_spec is defined %} + base_runtime_spec = "{{ containerd_cfg_dir }}/{{ runtime.base_runtime_spec }}" +{% endif %} + + [plugins."io.containerd.grpc.v1.cri".containerd.runtimes.{{ runtime.name }}.options] +{% for key, value in runtime.options.items() %} + {{ key }} = {{ value }} +{% endfor %} +{% endfor %} +{% if kata_containers_enabled %} + [plugins."io.containerd.grpc.v1.cri".containerd.runtimes.kata-qemu] + runtime_type = "io.containerd.kata-qemu.v2" +{% endif %} +{% if gvisor_enabled %} + [plugins."io.containerd.grpc.v1.cri".containerd.runtimes.runsc] + runtime_type = "io.containerd.runsc.v1" +{% endif %} + [plugins."io.containerd.grpc.v1.cri".registry] +{% if containerd_use_config_path is defined and containerd_use_config_path|bool %} + config_path = "{{ containerd_cfg_dir }}/certs.d" +{% else %} + [plugins."io.containerd.grpc.v1.cri".registry.mirrors] +{% set insecure_registries_addr = [] %} +{% for registry in containerd_registries_mirrors %} + [plugins."io.containerd.grpc.v1.cri".registry.mirrors."{{ registry.prefix }}"] +{% set endpoint = [] %} +{% for mirror in registry.mirrors %} +{% if endpoint.append(mirror.host) %}{% endif %} +{% if mirror.skip_verify is defined and mirror.skip_verify|bool %}{% if insecure_registries_addr.append(mirror.host | urlsplit('netloc')) %}{% endif %}{% endif %} +{% endfor %} + endpoint = ["{{ ( endpoint | unique ) | join('","') }}"] +{% endfor %} +{% for addr in insecure_registries_addr | unique %} + [plugins."io.containerd.grpc.v1.cri".registry.configs."{{ addr }}".tls] + insecure_skip_verify = true +{% endfor %} +{% endif %} +{% for registry in containerd_registry_auth if registry['registry'] is defined %} +{% if (registry['username'] is defined and registry['password'] is defined) or registry['auth'] is defined %} + [plugins."io.containerd.grpc.v1.cri".registry.configs."{{ registry['registry'] }}".auth] +{% if registry['username'] is defined and registry['password'] is defined %} + password = "{{ registry['password'] }}" + username = "{{ registry['username'] }}" +{% else %} + auth = "{{ registry['auth'] }}" +{% endif %} +{% endif %} +{% endfor %} + +{% if nri_enabled and containerd_version >= 1.7.0 %} + [plugins."io.containerd.nri.v1.nri"] + disable = false +{% endif %} + +{% if containerd_extra_args is defined %} +{{ containerd_extra_args }} +{% endif %} diff --git a/kubespray/roles/container-engine/containerd/templates/containerd.service.j2 b/kubespray/roles/container-engine/containerd/templates/containerd.service.j2 new file mode 100644 index 0000000..06b2290 --- /dev/null +++ b/kubespray/roles/container-engine/containerd/templates/containerd.service.j2 @@ -0,0 +1,45 @@ +# Copyright The containerd Authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +[Unit] +Description=containerd container runtime +Documentation=https://containerd.io +After=network.target local-fs.target + +[Service] +ExecStartPre=-/sbin/modprobe overlay +ExecStart={{ containerd_bin_dir }}/containerd + +Type=notify +Delegate=yes +KillMode=process +Restart=always +RestartSec=5 +# Having non-zero Limit*s causes performance problems due to accounting overhead +# in the kernel. We recommend using cgroups to do container-local accounting. +LimitNPROC={{ containerd_limit_proc_num }} +LimitCORE={{ containerd_limit_core }} +LimitNOFILE={{ containerd_limit_open_file_num }} +LimitMEMLOCK={{ containerd_limit_mem_lock }} +# Comment TasksMax if your systemd version does not supports it. +# Only systemd 226 and above support this version. +TasksMax=infinity +OOMScoreAdjust=-999 +# Set the cgroup slice of the service so that kube reserved takes effect +{% if kube_reserved is defined and kube_reserved|bool %} +Slice={{ kube_reserved_cgroups_for_service_slice }} +{% endif %} + +[Install] +WantedBy=multi-user.target diff --git a/kubespray/roles/container-engine/containerd/templates/hosts.toml.j2 b/kubespray/roles/container-engine/containerd/templates/hosts.toml.j2 new file mode 100644 index 0000000..c04dc47 --- /dev/null +++ b/kubespray/roles/container-engine/containerd/templates/hosts.toml.j2 @@ -0,0 +1,8 @@ +server = "https://{{ item.prefix }}" +{% for mirror in item.mirrors %} +[host."{{ mirror.host }}"] + capabilities = ["{{ ([ mirror.capabilities ] | flatten ) | join('","') }}"] +{% if mirror.skip_verify is defined %} + skip_verify = {{ mirror.skip_verify | default('false') | string | lower }} +{% endif %} +{% endfor %} diff --git a/kubespray/roles/container-engine/containerd/templates/http-proxy.conf.j2 b/kubespray/roles/container-engine/containerd/templates/http-proxy.conf.j2 new file mode 100644 index 0000000..212f30f --- /dev/null +++ b/kubespray/roles/container-engine/containerd/templates/http-proxy.conf.j2 @@ -0,0 +1,2 @@ +[Service] +Environment={% if http_proxy is defined %}"HTTP_PROXY={{ http_proxy }}"{% endif %} {% if https_proxy is defined %}"HTTPS_PROXY={{ https_proxy }}"{% endif %} {% if no_proxy is defined %}"NO_PROXY={{ no_proxy }}"{% endif %} diff --git a/kubespray/roles/container-engine/containerd/vars/debian.yml b/kubespray/roles/container-engine/containerd/vars/debian.yml new file mode 100644 index 0000000..8b18d9a --- /dev/null +++ b/kubespray/roles/container-engine/containerd/vars/debian.yml @@ -0,0 +1,7 @@ +--- +containerd_repo_info: + repos: + - > + deb {{ containerd_debian_repo_base_url }} + {{ ansible_distribution_release | lower }} + {{ containerd_debian_repo_component }} diff --git a/kubespray/roles/container-engine/containerd/vars/ubuntu.yml b/kubespray/roles/container-engine/containerd/vars/ubuntu.yml new file mode 100644 index 0000000..dd77532 --- /dev/null +++ b/kubespray/roles/container-engine/containerd/vars/ubuntu.yml @@ -0,0 +1,7 @@ +--- +containerd_repo_info: + repos: + - > + deb {{ containerd_ubuntu_repo_base_url }} + {{ ansible_distribution_release | lower }} + {{ containerd_ubuntu_repo_component }} diff --git a/kubespray/roles/container-engine/cri-dockerd/handlers/main.yml b/kubespray/roles/container-engine/cri-dockerd/handlers/main.yml new file mode 100644 index 0000000..3990d33 --- /dev/null +++ b/kubespray/roles/container-engine/cri-dockerd/handlers/main.yml @@ -0,0 +1,35 @@ +--- +- name: Restart and enable cri-dockerd + command: /bin/true + notify: + - Cri-dockerd | reload systemd + - Cri-dockerd | restart docker.service + - Cri-dockerd | reload cri-dockerd.socket + - Cri-dockerd | reload cri-dockerd.service + - Cri-dockerd | enable cri-dockerd service + +- name: Cri-dockerd | reload systemd + systemd: + name: cri-dockerd + daemon_reload: true + masked: no + +- name: Cri-dockerd | restart docker.service + service: + name: docker.service + state: restarted + +- name: Cri-dockerd | reload cri-dockerd.socket + service: + name: cri-dockerd.socket + state: restarted + +- name: Cri-dockerd | reload cri-dockerd.service + service: + name: cri-dockerd.service + state: restarted + +- name: Cri-dockerd | enable cri-dockerd service + service: + name: cri-dockerd.service + enabled: yes diff --git a/kubespray/roles/container-engine/cri-dockerd/meta/main.yml b/kubespray/roles/container-engine/cri-dockerd/meta/main.yml new file mode 100644 index 0000000..4923f3b --- /dev/null +++ b/kubespray/roles/container-engine/cri-dockerd/meta/main.yml @@ -0,0 +1,4 @@ +--- +dependencies: + - role: container-engine/docker + - role: container-engine/crictl diff --git a/kubespray/roles/container-engine/cri-dockerd/molecule/default/converge.yml b/kubespray/roles/container-engine/cri-dockerd/molecule/default/converge.yml new file mode 100644 index 0000000..be6fa38 --- /dev/null +++ b/kubespray/roles/container-engine/cri-dockerd/molecule/default/converge.yml @@ -0,0 +1,9 @@ +--- +- name: Converge + hosts: all + become: true + vars: + container_manager: docker + roles: + - role: kubespray-defaults + - role: container-engine/cri-dockerd diff --git a/kubespray/roles/container-engine/cri-dockerd/molecule/default/files/10-mynet.conf b/kubespray/roles/container-engine/cri-dockerd/molecule/default/files/10-mynet.conf new file mode 100644 index 0000000..f10935b --- /dev/null +++ b/kubespray/roles/container-engine/cri-dockerd/molecule/default/files/10-mynet.conf @@ -0,0 +1,17 @@ +{ + "cniVersion": "0.2.0", + "name": "mynet", + "type": "bridge", + "bridge": "cni0", + "isGateway": true, + "ipMasq": true, + "ipam": { + "type": "host-local", + "subnet": "172.19.0.0/24", + "routes": [ + { + "dst": "0.0.0.0/0" + } + ] + } +} diff --git a/kubespray/roles/container-engine/cri-dockerd/molecule/default/files/container.json b/kubespray/roles/container-engine/cri-dockerd/molecule/default/files/container.json new file mode 100644 index 0000000..1d839e6 --- /dev/null +++ b/kubespray/roles/container-engine/cri-dockerd/molecule/default/files/container.json @@ -0,0 +1,10 @@ +{ + "metadata": { + "name": "cri-dockerd1" + }, + "image": { + "image": "quay.io/kubespray/hello-world:latest" + }, + "log_path": "cri-dockerd1.0.log", + "linux": {} +} diff --git a/kubespray/roles/container-engine/cri-dockerd/molecule/default/files/sandbox.json b/kubespray/roles/container-engine/cri-dockerd/molecule/default/files/sandbox.json new file mode 100644 index 0000000..f451e9e --- /dev/null +++ b/kubespray/roles/container-engine/cri-dockerd/molecule/default/files/sandbox.json @@ -0,0 +1,10 @@ +{ + "metadata": { + "name": "cri-dockerd1", + "namespace": "default", + "attempt": 1, + "uid": "hdishd83djaidwnduwk28bcsb" + }, + "linux": {}, + "log_directory": "/tmp" +} diff --git a/kubespray/roles/container-engine/cri-dockerd/molecule/default/molecule.yml b/kubespray/roles/container-engine/cri-dockerd/molecule/default/molecule.yml new file mode 100644 index 0000000..82cb778 --- /dev/null +++ b/kubespray/roles/container-engine/cri-dockerd/molecule/default/molecule.yml @@ -0,0 +1,39 @@ +--- +role_name_check: 1 +driver: + name: vagrant + provider: + name: libvirt +platforms: + - name: almalinux8 + box: almalinux/8 + cpus: 1 + memory: 1024 + nested: true + groups: + - kube_control_plane + provider_options: + driver: kvm + - name: ubuntu20 + box: generic/ubuntu2004 + cpus: 1 + memory: 1024 + nested: true + groups: + - kube_control_plane + provider_options: + driver: kvm +provisioner: + name: ansible + env: + ANSIBLE_ROLES_PATH: ../../../../ + config_options: + defaults: + callbacks_enabled: profile_tasks + timeout: 120 + inventory: + group_vars: + all: + become: true +verifier: + name: testinfra diff --git a/kubespray/roles/container-engine/cri-dockerd/molecule/default/prepare.yml b/kubespray/roles/container-engine/cri-dockerd/molecule/default/prepare.yml new file mode 100644 index 0000000..83449f8 --- /dev/null +++ b/kubespray/roles/container-engine/cri-dockerd/molecule/default/prepare.yml @@ -0,0 +1,48 @@ +--- +- name: Prepare + hosts: all + become: true + roles: + - role: kubespray-defaults + - role: bootstrap-os + - role: adduser + user: "{{ addusers.kube }}" + tasks: + - name: Download CNI + include_tasks: "../../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.cni) }}" + +- name: Prepare container runtime + hosts: all + become: true + vars: + container_manager: containerd + kube_network_plugin: cni + roles: + - role: kubespray-defaults + - role: network_plugin/cni + tasks: + - name: Copy test container files + copy: + src: "{{ item }}" + dest: "/tmp/{{ item }}" + owner: root + mode: 0644 + with_items: + - container.json + - sandbox.json + - name: Create /etc/cni/net.d directory + file: + path: /etc/cni/net.d + state: directory + owner: "{{ kube_owner }}" + mode: 0755 + - name: Setup CNI + copy: + src: "{{ item }}" + dest: "/etc/cni/net.d/{{ item }}" + owner: root + mode: 0644 + with_items: + - 10-mynet.conf diff --git a/kubespray/roles/container-engine/cri-dockerd/molecule/default/tests/test_default.py b/kubespray/roles/container-engine/cri-dockerd/molecule/default/tests/test_default.py new file mode 100644 index 0000000..dc99b34 --- /dev/null +++ b/kubespray/roles/container-engine/cri-dockerd/molecule/default/tests/test_default.py @@ -0,0 +1,19 @@ +import os + +import testinfra.utils.ansible_runner + +testinfra_hosts = testinfra.utils.ansible_runner.AnsibleRunner( + os.environ['MOLECULE_INVENTORY_FILE']).get_hosts('all') + + +def test_run_pod(host): + run_command = "/usr/local/bin/crictl run --with-pull /tmp/container.json /tmp/sandbox.json" + with host.sudo(): + cmd = host.command(run_command) + assert cmd.rc == 0 + + with host.sudo(): + log_f = host.file("/tmp/cri-dockerd1.0.log") + + assert log_f.exists + assert b"Hello from Docker" in log_f.content diff --git a/kubespray/roles/container-engine/cri-dockerd/tasks/main.yml b/kubespray/roles/container-engine/cri-dockerd/tasks/main.yml new file mode 100644 index 0000000..f8965fd --- /dev/null +++ b/kubespray/roles/container-engine/cri-dockerd/tasks/main.yml @@ -0,0 +1,28 @@ +--- +- name: Runc | Download cri-dockerd binary + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.cri_dockerd) }}" + +- name: Copy cri-dockerd binary from download dir + copy: + src: "{{ local_release_dir }}/cri-dockerd" + dest: "{{ bin_dir }}/cri-dockerd" + mode: 0755 + remote_src: true + notify: + - Restart and enable cri-dockerd + +- name: Generate cri-dockerd systemd unit files + template: + src: "{{ item }}.j2" + dest: "/etc/systemd/system/{{ item }}" + mode: 0644 + with_items: + - cri-dockerd.service + - cri-dockerd.socket + notify: + - Restart and enable cri-dockerd + +- name: Flush handlers + meta: flush_handlers diff --git a/kubespray/roles/container-engine/cri-dockerd/templates/cri-dockerd.service.j2 b/kubespray/roles/container-engine/cri-dockerd/templates/cri-dockerd.service.j2 new file mode 100644 index 0000000..ec12815 --- /dev/null +++ b/kubespray/roles/container-engine/cri-dockerd/templates/cri-dockerd.service.j2 @@ -0,0 +1,44 @@ +[Unit] +Description=CRI Interface for Docker Application Container Engine +Documentation=https://docs.mirantis.com +After=network-online.target firewalld.service docker.service +Wants=network-online.target docker.service +Requires=cri-dockerd.socket + +[Service] +Type=notify +ExecStart={{ bin_dir }}/cri-dockerd --container-runtime-endpoint {{ cri_socket }} --cni-conf-dir=/etc/cni/net.d --cni-bin-dir=/opt/cni/bin --network-plugin=cni --pod-cidr={{ kube_pods_subnet }} --pod-infra-container-image={{ pod_infra_image_repo }}:{{ pod_infra_version }} {% if enable_dual_stack_networks %}--ipv6-dual-stack=True{% endif %} + +ExecReload=/bin/kill -s HUP $MAINPID +TimeoutSec=0 +RestartSec=2 +Restart=always + +# Note that StartLimit* options were moved from "Service" to "Unit" in systemd 229. +# Both the old, and new location are accepted by systemd 229 and up, so using the old location +# to make them work for either version of systemd. +StartLimitBurst=3 + +# Note that StartLimitInterval was renamed to StartLimitIntervalSec in systemd 230. +# Both the old, and new name are accepted by systemd 230 and up, so using the old name to make +# this option work for either version of systemd. +StartLimitInterval=60s + +# Having non-zero Limit*s causes performance problems due to accounting overhead +# in the kernel. We recommend using cgroups to do container-local accounting. +LimitNOFILE=infinity +LimitNPROC=infinity +LimitCORE=infinity + +# Comment TasksMax if your systemd version does not support it. +# Only systemd 226 and above support this option. +TasksMax=infinity +Delegate=yes +KillMode=process +# Set the cgroup slice of the service so that kube reserved takes effect +{% if kube_reserved is defined and kube_reserved|bool %} +Slice={{ kube_reserved_cgroups_for_service_slice }} +{% endif %} + +[Install] +WantedBy=multi-user.target diff --git a/kubespray/roles/container-engine/cri-dockerd/templates/cri-dockerd.socket.j2 b/kubespray/roles/container-engine/cri-dockerd/templates/cri-dockerd.socket.j2 new file mode 100644 index 0000000..8dfa27d --- /dev/null +++ b/kubespray/roles/container-engine/cri-dockerd/templates/cri-dockerd.socket.j2 @@ -0,0 +1,12 @@ +[Unit] +Description=CRI Docker Socket for the API +PartOf=cri-dockerd.service + +[Socket] +ListenStream=%t/cri-dockerd.sock +SocketMode=0660 +SocketUser=root +SocketGroup=docker + +[Install] +WantedBy=sockets.target diff --git a/kubespray/roles/container-engine/cri-o/defaults/main.yml b/kubespray/roles/container-engine/cri-o/defaults/main.yml new file mode 100644 index 0000000..949ed69 --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/defaults/main.yml @@ -0,0 +1,99 @@ +--- + +crio_cgroup_manager: "{{ kubelet_cgroup_driver | default('systemd') }}" +crio_conmon: "{{ bin_dir }}/conmon" +crio_enable_metrics: false +crio_log_level: "info" +crio_metrics_port: "9090" +crio_pause_image: "{{ pod_infra_image_repo }}:{{ pod_infra_version }}" + +# Registries defined within cri-o. +# By default unqualified images are not allowed for security reasons +crio_registries: [] +# - prefix: docker.io +# insecure: false +# blocked: false +# location: registry-1.docker.io ## REQUIRED +# unqualified: false +# mirrors: +# - location: 172.20.100.52:5000 +# insecure: true +# - location: mirror.gcr.io +# insecure: false + +crio_registry_auth: [] +# - registry: 10.0.0.2:5000 +# username: user +# password: pass + +crio_seccomp_profile: "" +crio_selinux: "{{ (preinstall_selinux_state == 'enforcing') | lower }}" +crio_signature_policy: "{% if ansible_os_family == 'ClearLinux' %}/usr/share/defaults/crio/policy.json{% endif %}" + +# Override system default for storage driver +# crio_storage_driver: "overlay" + +crio_stream_port: "10010" + +crio_required_version: "{{ kube_version | regex_replace('^v(?P\\d+).(?P\\d+).(?P\\d+)$', '\\g.\\g') }}" + +# The crio_runtimes variable defines a list of OCI compatible runtimes. +crio_runtimes: + - name: runc + path: "{{ bin_dir }}/runc" + type: oci + root: /run/runc + +# Kata Containers is an OCI runtime, where containers are run inside lightweight +# VMs. Kata provides additional isolation towards the host, minimizing the host attack +# surface and mitigating the consequences of containers breakout. +kata_runtimes: + # Kata Containers with the default configured VMM + - name: kata-qemu + path: /usr/local/bin/containerd-shim-kata-qemu-v2 + type: vm + root: /run/kata-containers + privileged_without_host_devices: true + +# crun is a fast and low-memory footprint OCI Container Runtime fully written in C. +crun_runtime: + name: crun + path: "{{ bin_dir }}/crun" + type: oci + root: /run/crun + +# youki is an implementation of the OCI runtime-spec in Rust, similar to runc. +youki_runtime: + name: youki + path: "{{ youki_bin_dir }}/youki" + type: oci + root: /run/youki + +# TODO(cristicalin): remove this after 2.21 +crio_download_base: "download.opensuse.org/repositories/devel:kubic:libcontainers:stable" +crio_download_crio: "http://{{ crio_download_base }}:/cri-o:/" + +# Reserve 16M uids and gids for user namespaces (256 pods * 65536 uids/gids) +# at the end of the uid/gid space +crio_remap_enable: false +crio_remap_user: containers +crio_subuid_start: 2130706432 +crio_subuid_length: 16777216 +crio_subgid_start: 2130706432 +crio_subgid_length: 16777216 + +# cri-o binary files +crio_bin_files: + - conmon + - crio + - crio-status + - pinns + +# cri-o manual files +crio_man_files: + 5: + - crio.conf + - crio.conf.d + 8: + - crio + - crio-status diff --git a/kubespray/roles/container-engine/cri-o/files/mounts.conf b/kubespray/roles/container-engine/cri-o/files/mounts.conf new file mode 100644 index 0000000..b7cde9d --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/files/mounts.conf @@ -0,0 +1 @@ +/usr/share/rhel/secrets:/run/secrets diff --git a/kubespray/roles/container-engine/cri-o/handlers/main.yml b/kubespray/roles/container-engine/cri-o/handlers/main.yml new file mode 100644 index 0000000..763f4b5 --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/handlers/main.yml @@ -0,0 +1,16 @@ +--- +- name: Restart crio + command: /bin/true + notify: + - CRI-O | reload systemd + - CRI-O | reload crio + +- name: CRI-O | reload systemd + systemd: + daemon_reload: true + +- name: CRI-O | reload crio + service: + name: crio + state: restarted + enabled: yes diff --git a/kubespray/roles/container-engine/cri-o/meta/main.yml b/kubespray/roles/container-engine/cri-o/meta/main.yml new file mode 100644 index 0000000..7259b46 --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/meta/main.yml @@ -0,0 +1,5 @@ +--- +dependencies: + - role: container-engine/runc + - role: container-engine/crictl + - role: container-engine/skopeo diff --git a/kubespray/roles/container-engine/cri-o/molecule/default/converge.yml b/kubespray/roles/container-engine/cri-o/molecule/default/converge.yml new file mode 100644 index 0000000..376f07c --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/molecule/default/converge.yml @@ -0,0 +1,9 @@ +--- +- name: Converge + hosts: all + become: true + vars: + container_manager: crio + roles: + - role: kubespray-defaults + - role: container-engine/cri-o diff --git a/kubespray/roles/container-engine/cri-o/molecule/default/files/10-mynet.conf b/kubespray/roles/container-engine/cri-o/molecule/default/files/10-mynet.conf new file mode 100644 index 0000000..f10935b --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/molecule/default/files/10-mynet.conf @@ -0,0 +1,17 @@ +{ + "cniVersion": "0.2.0", + "name": "mynet", + "type": "bridge", + "bridge": "cni0", + "isGateway": true, + "ipMasq": true, + "ipam": { + "type": "host-local", + "subnet": "172.19.0.0/24", + "routes": [ + { + "dst": "0.0.0.0/0" + } + ] + } +} diff --git a/kubespray/roles/container-engine/cri-o/molecule/default/files/container.json b/kubespray/roles/container-engine/cri-o/molecule/default/files/container.json new file mode 100644 index 0000000..bcd71e7 --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/molecule/default/files/container.json @@ -0,0 +1,10 @@ +{ + "metadata": { + "name": "runc1" + }, + "image": { + "image": "quay.io/kubespray/hello-world:latest" + }, + "log_path": "runc1.0.log", + "linux": {} +} diff --git a/kubespray/roles/container-engine/cri-o/molecule/default/files/sandbox.json b/kubespray/roles/container-engine/cri-o/molecule/default/files/sandbox.json new file mode 100644 index 0000000..eb9dcb9 --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/molecule/default/files/sandbox.json @@ -0,0 +1,10 @@ +{ + "metadata": { + "name": "runc1", + "namespace": "default", + "attempt": 1, + "uid": "hdishd83djaidwnduwk28bcsb" + }, + "linux": {}, + "log_directory": "/tmp" +} diff --git a/kubespray/roles/container-engine/cri-o/molecule/default/molecule.yml b/kubespray/roles/container-engine/cri-o/molecule/default/molecule.yml new file mode 100644 index 0000000..99d44a3 --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/molecule/default/molecule.yml @@ -0,0 +1,57 @@ +--- +role_name_check: 1 +driver: + name: vagrant + provider: + name: libvirt +platforms: + - name: ubuntu20 + box: generic/ubuntu2004 + cpus: 2 + memory: 1024 + groups: + - kube_control_plane + - kube_node + - k8s_cluster + provider_options: + driver: kvm + - name: almalinux8 + box: almalinux/8 + cpus: 2 + memory: 1024 + groups: + - kube_control_plane + - kube_node + - k8s_cluster + provider_options: + driver: kvm + - name: fedora + box: fedora/38-cloud-base + cpus: 2 + memory: 2048 + groups: + - kube_control_plane + - kube_node + - k8s_cluster + provider_options: + driver: kvm + - name: debian10 + box: generic/debian10 + cpus: 2 + memory: 1024 + groups: + - kube_control_plane + - kube_node + - k8s_cluster + provider_options: + driver: kvm +provisioner: + name: ansible + env: + ANSIBLE_ROLES_PATH: ../../../../ + config_options: + defaults: + callbacks_enabled: profile_tasks + timeout: 120 +verifier: + name: testinfra diff --git a/kubespray/roles/container-engine/cri-o/molecule/default/prepare.yml b/kubespray/roles/container-engine/cri-o/molecule/default/prepare.yml new file mode 100644 index 0000000..103b0d3 --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/molecule/default/prepare.yml @@ -0,0 +1,53 @@ +--- +- name: Prepare + hosts: all + gather_facts: False + become: true + vars: + ignore_assert_errors: true + roles: + - role: kubespray-defaults + - role: bootstrap-os + - role: kubernetes/preinstall + - role: adduser + user: "{{ addusers.kube }}" + tasks: + - name: Download CNI + include_tasks: "../../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.cni) }}" + +- name: Prepare CNI + hosts: all + gather_facts: False + become: true + vars: + ignore_assert_errors: true + kube_network_plugin: cni + roles: + - role: kubespray-defaults + - role: network_plugin/cni + tasks: + - name: Copy test container files + copy: + src: "{{ item }}" + dest: "/tmp/{{ item }}" + owner: root + mode: 0644 + with_items: + - container.json + - sandbox.json + - name: Create /etc/cni/net.d directory + file: + path: /etc/cni/net.d + state: directory + owner: "{{ kube_owner }}" + mode: 0755 + - name: Setup CNI + copy: + src: "{{ item }}" + dest: "/etc/cni/net.d/{{ item }}" + owner: root + mode: 0644 + with_items: + - 10-mynet.conf diff --git a/kubespray/roles/container-engine/cri-o/molecule/default/tests/test_default.py b/kubespray/roles/container-engine/cri-o/molecule/default/tests/test_default.py new file mode 100644 index 0000000..358a1b7 --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/molecule/default/tests/test_default.py @@ -0,0 +1,35 @@ +import os + +import testinfra.utils.ansible_runner + +testinfra_hosts = testinfra.utils.ansible_runner.AnsibleRunner( + os.environ['MOLECULE_INVENTORY_FILE']).get_hosts('all') + + +def test_service(host): + svc = host.service("crio") + assert svc.is_running + assert svc.is_enabled + + +def test_run(host): + crictl = "/usr/local/bin/crictl" + path = "unix:///var/run/crio/crio.sock" + with host.sudo(): + cmd = host.command(crictl + " --runtime-endpoint " + path + " version") + assert cmd.rc == 0 + assert "RuntimeName: cri-o" in cmd.stdout + +def test_run_pod(host): + runtime = "runc" + + run_command = "/usr/local/bin/crictl run --with-pull --runtime {} /tmp/container.json /tmp/sandbox.json".format(runtime) + with host.sudo(): + cmd = host.command(run_command) + assert cmd.rc == 0 + + with host.sudo(): + log_f = host.file("/tmp/runc1.0.log") + + assert log_f.exists + assert b"Hello from Docker" in log_f.content diff --git a/kubespray/roles/container-engine/cri-o/tasks/cleanup.yaml b/kubespray/roles/container-engine/cri-o/tasks/cleanup.yaml new file mode 100644 index 0000000..2b8251c --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/tasks/cleanup.yaml @@ -0,0 +1,120 @@ +--- +# TODO(cristicalin): drop this file after 2.21 +- name: CRI-O kubic repo name for debian os family + set_fact: + crio_kubic_debian_repo_name: "{{ ((ansible_distribution == 'Ubuntu') | ternary('x', '')) ~ ansible_distribution ~ '_' ~ ansible_distribution_version }}" + when: ansible_os_family == "Debian" + +- name: Remove legacy CRI-O kubic apt repo key + apt_key: + url: "https://{{ crio_download_base }}/{{ crio_kubic_debian_repo_name }}/Release.key" + state: absent + environment: "{{ proxy_env }}" + when: crio_kubic_debian_repo_name is defined + +- name: Remove legacy CRI-O kubic apt repo + apt_repository: + repo: "deb http://{{ crio_download_base }}/{{ crio_kubic_debian_repo_name }}/ /" + state: absent + filename: devel-kubic-libcontainers-stable + when: crio_kubic_debian_repo_name is defined + +- name: Remove legacy CRI-O kubic cri-o apt repo + apt_repository: + repo: "deb {{ crio_download_crio }}{{ crio_version }}/{{ crio_kubic_debian_repo_name }}/ /" + state: absent + filename: devel-kubic-libcontainers-stable-cri-o + when: crio_kubic_debian_repo_name is defined + +- name: Remove legacy CRI-O kubic yum repo + yum_repository: + name: devel_kubic_libcontainers_stable + description: Stable Releases of Upstream github.com/containers packages (CentOS_$releasever) + baseurl: http://{{ crio_download_base }}/CentOS_{{ ansible_distribution_major_version }}/ + state: absent + when: + - ansible_os_family == "RedHat" + - ansible_distribution not in ["Amazon", "Fedora"] + +- name: Remove legacy CRI-O kubic yum repo + yum_repository: + name: "devel_kubic_libcontainers_stable_cri-o_{{ crio_version }}" + description: "CRI-O {{ crio_version }} (CentOS_$releasever)" + baseurl: "{{ crio_download_crio }}{{ crio_version }}/CentOS_{{ ansible_distribution_major_version }}/" + state: absent + when: + - ansible_os_family == "RedHat" + - ansible_distribution not in ["Amazon", "Fedora"] + +- name: Remove legacy CRI-O kubic yum repo + yum_repository: + name: devel_kubic_libcontainers_stable + description: Stable Releases of Upstream github.com/containers packages + baseurl: http://{{ crio_download_base }}/Fedora_{{ ansible_distribution_major_version }}/ + state: absent + when: + - ansible_distribution in ["Fedora"] + - not is_ostree + +- name: Remove legacy CRI-O kubic yum repo + yum_repository: + name: "devel_kubic_libcontainers_stable_cri-o_{{ crio_version }}" + description: "CRI-O {{ crio_version }}" + baseurl: "{{ crio_download_crio }}{{ crio_version }}/Fedora_{{ ansible_distribution_major_version }}/" + state: absent + when: + - ansible_distribution in ["Fedora"] + - not is_ostree + +- name: Remove legacy CRI-O kubic yum repo + yum_repository: + name: devel_kubic_libcontainers_stable + description: Stable Releases of Upstream github.com/containers packages + baseurl: http://{{ crio_download_base }}/CentOS_7/ + state: absent + when: ansible_distribution in ["Amazon"] + +- name: Remove legacy CRI-O kubic yum repo + yum_repository: + name: "devel_kubic_libcontainers_stable_cri-o_{{ crio_version }}" + description: "CRI-O {{ crio_version }}" + baseurl: "{{ crio_download_crio }}{{ crio_version }}/CentOS_7/" + state: absent + when: ansible_distribution in ["Amazon"] + +- name: Disable modular repos for CRI-O + community.general.ini_file: + path: "/etc/yum.repos.d/{{ item.repo }}.repo" + section: "{{ item.section }}" + option: enabled + value: 0 + mode: 0644 + become: true + when: is_ostree + loop: + - repo: "fedora-updates-modular" + section: "updates-modular" + - repo: "fedora-modular" + section: "fedora-modular" + +# Disable any older module version if we enabled them before +- name: Disable CRI-O ex module + command: "rpm-ostree ex module disable cri-o:{{ item }}" + become: true + when: + - is_ostree + - ostree_version is defined and ostree_version.stdout is version('2021.9', '>=') + with_items: + - 1.22 + - 1.23 + - 1.24 + +- name: Cri-o | remove installed packages + package: + name: "{{ item }}" + state: absent + when: not is_ostree + with_items: + - cri-o + - cri-o-runc + - oci-systemd-hook diff --git a/kubespray/roles/container-engine/cri-o/tasks/main.yaml b/kubespray/roles/container-engine/cri-o/tasks/main.yaml new file mode 100644 index 0000000..f5df974 --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/tasks/main.yaml @@ -0,0 +1,214 @@ +--- +- name: Cri-o | check if fedora coreos + stat: + path: /run/ostree-booted + get_attributes: no + get_checksum: no + get_mime: no + register: ostree + +- name: Cri-o | set is_ostree + set_fact: + is_ostree: "{{ ostree.stat.exists }}" + +- name: Cri-o | get ostree version + shell: "set -o pipefail && rpm-ostree --version | awk -F\\' '/Version/{print $2}'" + args: + executable: /bin/bash + register: ostree_version + when: is_ostree + +- name: Cri-o | Download cri-o + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.crio) }}" + +- name: Cri-o | special handling for amazon linux + import_tasks: "setup-amazon.yaml" + when: ansible_distribution in ["Amazon"] + +- name: Cri-o | clean up reglacy repos + import_tasks: "cleanup.yaml" + +- name: Cri-o | build a list of crio runtimes with Katacontainers runtimes + set_fact: + crio_runtimes: "{{ crio_runtimes + kata_runtimes }}" + when: + - kata_containers_enabled + +- name: Cri-o | build a list of crio runtimes with crun runtime + set_fact: + crio_runtimes: "{{ crio_runtimes + [crun_runtime] }}" + when: + - crun_enabled + +- name: Cri-o | build a list of crio runtimes with youki runtime + set_fact: + crio_runtimes: "{{ crio_runtimes + [youki_runtime] }}" + when: + - youki_enabled + +- name: Cri-o | make sure needed folders exist in the system + with_items: + - /etc/crio + - /etc/containers + - /etc/systemd/system/crio.service.d + file: + path: "{{ item }}" + state: directory + mode: 0755 + +- name: Cri-o | install cri-o config + template: + src: crio.conf.j2 + dest: /etc/crio/crio.conf + mode: 0644 + register: config_install + +- name: Cri-o | install config.json + template: + src: config.json.j2 + dest: /etc/crio/config.json + mode: 0644 + register: reg_auth_install + +- name: Cri-o | copy binaries + copy: + src: "{{ local_release_dir }}/cri-o/bin/{{ item }}" + dest: "{{ bin_dir }}/{{ item }}" + mode: 0755 + remote_src: true + with_items: + - "{{ crio_bin_files }}" + notify: Restart crio + +- name: Cri-o | copy service file + copy: + src: "{{ local_release_dir }}/cri-o/contrib/crio.service" + dest: /etc/systemd/system/crio.service + mode: 0755 + remote_src: true + notify: Restart crio + +- name: Cri-o | update the bin dir for crio.service file + replace: + dest: /etc/systemd/system/crio.service + regexp: "/usr/local/bin/crio" + replace: "{{ bin_dir }}/crio" + notify: Restart crio + +- name: Cri-o | copy default policy + copy: + src: "{{ local_release_dir }}/cri-o/contrib/policy.json" + dest: /etc/containers/policy.json + mode: 0755 + remote_src: true + notify: Restart crio + +- name: Cri-o | copy mounts.conf + copy: + src: mounts.conf + dest: /etc/containers/mounts.conf + mode: 0644 + when: + - ansible_os_family == 'RedHat' + notify: Restart crio + +- name: Cri-o | create directory for oci hooks + file: + path: /etc/containers/oci/hooks.d + state: directory + owner: root + mode: 0755 + +- name: Cri-o | set overlay driver + community.general.ini_file: + dest: /etc/containers/storage.conf + section: storage + option: "{{ item.option }}" + value: "{{ item.value }}" + mode: 0644 + with_items: + - option: driver + value: '"overlay"' + - option: graphroot + value: '"/var/lib/containers/storage"' + - option: runroot + value: '"/var/run/containers/storage"' + +# metacopy=on is available since 4.19 and was backported to RHEL 4.18 kernel +- name: Cri-o | set metacopy mount options correctly + community.general.ini_file: + dest: /etc/containers/storage.conf + section: storage.options.overlay + option: mountopt + value: '{{ ''"nodev"'' if ansible_kernel is version_compare(("4.18" if ansible_os_family == "RedHat" else "4.19"), "<") else ''"nodev,metacopy=on"'' }}' + mode: 0644 + +- name: Cri-o | create directory registries configs + file: + path: /etc/containers/registries.conf.d + state: directory + owner: root + mode: 0755 + +- name: Cri-o | write registries configs + template: + src: registry.conf.j2 + dest: "/etc/containers/registries.conf.d/10-{{ item.prefix | default(item.location) | regex_replace(':', '_') }}.conf" + mode: 0644 + loop: "{{ crio_registries }}" + notify: Restart crio + +- name: Cri-o | configure unqualified registry settings + template: + src: unqualified.conf.j2 + dest: "/etc/containers/registries.conf.d/01-unqualified.conf" + mode: 0644 + notify: Restart crio + +- name: Cri-o | write cri-o proxy drop-in + template: + src: http-proxy.conf.j2 + dest: /etc/systemd/system/crio.service.d/http-proxy.conf + mode: 0644 + notify: Restart crio + when: http_proxy is defined or https_proxy is defined + +- name: Cri-o | configure the uid/gid space for user namespaces + lineinfile: + path: '{{ item.path }}' + line: '{{ item.entry }}' + regex: '^\s*{{ crio_remap_user }}:' + state: '{{ "present" if crio_remap_enable | bool else "absent" }}' + loop: + - path: /etc/subuid + entry: '{{ crio_remap_user }}:{{ crio_subuid_start }}:{{ crio_subuid_length }}' + - path: /etc/subgid + entry: '{{ crio_remap_user }}:{{ crio_subgid_start }}:{{ crio_subgid_length }}' + loop_control: + label: '{{ item.path }}' + +- name: Cri-o | ensure crio service is started and enabled + service: + name: crio + daemon_reload: true + enabled: true + state: started + register: service_start + +- name: Cri-o | trigger service restart only when needed + service: + name: crio + state: restarted + when: + - config_install.changed or reg_auth_install.changed + - not service_start.changed + +- name: Cri-o | verify that crio is running + command: "{{ bin_dir }}/crio-status info" + register: get_crio_info + until: get_crio_info is succeeded + changed_when: false + retries: 5 + delay: "{{ retry_stagger | random + 3 }}" diff --git a/kubespray/roles/container-engine/cri-o/tasks/reset.yml b/kubespray/roles/container-engine/cri-o/tasks/reset.yml new file mode 100644 index 0000000..65ee002 --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/tasks/reset.yml @@ -0,0 +1,87 @@ +--- +- name: CRI-O | Kubic repo name for debian os family + set_fact: + crio_kubic_debian_repo_name: "{{ ((ansible_distribution == 'Ubuntu') | ternary('x', '')) ~ ansible_distribution ~ '_' ~ ansible_distribution_version }}" + when: ansible_os_family == "Debian" + tags: + - reset_crio + +- name: CRI-O | Remove kubic apt repo + apt_repository: + repo: "deb http://{{ crio_download_base }}/{{ crio_kubic_debian_repo_name }}/ /" + state: absent + when: crio_kubic_debian_repo_name is defined + tags: + - reset_crio + +- name: CRI-O | Remove cri-o apt repo + apt_repository: + repo: "deb {{ crio_download_crio }}{{ crio_version }}/{{ crio_kubic_debian_repo_name }}/ /" + state: present + filename: devel-kubic-libcontainers-stable-cri-o + when: crio_kubic_debian_repo_name is defined + tags: + - reset_crio + +- name: CRI-O | Remove CRI-O kubic yum repo + yum_repository: + name: devel_kubic_libcontainers_stable + state: absent + when: ansible_distribution in ["Amazon"] + tags: + - reset_crio + +- name: CRI-O | Remove CRI-O kubic yum repo + yum_repository: + name: "devel_kubic_libcontainers_stable_cri-o_{{ crio_version }}" + state: absent + when: + - ansible_os_family == "RedHat" + - ansible_distribution not in ["Amazon", "Fedora"] + tags: + - reset_crio + +- name: CRI-O | Run yum-clean-metadata + command: yum clean metadata + when: + - ansible_os_family == "RedHat" + tags: + - reset_crio + +- name: CRI-O | Remove crictl + file: + name: "{{ item }}" + state: absent + loop: + - /etc/crictl.yaml + - "{{ bin_dir }}/crictl" + tags: + - reset_crio + +- name: CRI-O | Stop crio service + service: + name: crio + daemon_reload: true + enabled: false + state: stopped + tags: + - reset_crio + +- name: CRI-O | Remove CRI-O configuration files + file: + name: "{{ item }}" + state: absent + loop: + - /etc/crio + - /etc/containers + - /etc/systemd/system/crio.service.d + tags: + - reset_crio + +- name: CRI-O | Remove CRI-O binaries + file: + name: "{{ item }}" + state: absent + with_items: "{{ crio_bin_files }}" + tags: + - reset_crio diff --git a/kubespray/roles/container-engine/cri-o/tasks/setup-amazon.yaml b/kubespray/roles/container-engine/cri-o/tasks/setup-amazon.yaml new file mode 100644 index 0000000..843bc20 --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/tasks/setup-amazon.yaml @@ -0,0 +1,38 @@ +--- +- name: Check that amzn2-extras.repo exists + stat: + path: /etc/yum.repos.d/amzn2-extras.repo + register: amzn2_extras_file_stat + +- name: Find docker repo in amzn2-extras.repo file + lineinfile: + dest: /etc/yum.repos.d/amzn2-extras.repo + line: "[amzn2extra-docker]" + check_mode: yes + register: amzn2_extras_docker_repo + when: + - amzn2_extras_file_stat.stat.exists + +- name: Remove docker repository + community.general.ini_file: + dest: /etc/yum.repos.d/amzn2-extras.repo + section: amzn2extra-docker + option: enabled + value: "0" + backup: yes + mode: 0644 + when: + - amzn2_extras_file_stat.stat.exists + - not amzn2_extras_docker_repo.changed + +- name: Add container-selinux yum repo + yum_repository: + name: copr:copr.fedorainfracloud.org:lsm5:container-selinux + file: _copr_lsm5-container-selinux.repo + description: Copr repo for container-selinux owned by lsm5 + baseurl: https://download.copr.fedorainfracloud.org/results/lsm5/container-selinux/epel-7-$basearch/ + gpgcheck: yes + gpgkey: https://download.copr.fedorainfracloud.org/results/lsm5/container-selinux/pubkey.gpg + skip_if_unavailable: yes + enabled: yes + repo_gpgcheck: no diff --git a/kubespray/roles/container-engine/cri-o/templates/config.json.j2 b/kubespray/roles/container-engine/cri-o/templates/config.json.j2 new file mode 100644 index 0000000..4afd49f --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/templates/config.json.j2 @@ -0,0 +1,17 @@ +{% if crio_registry_auth is defined and crio_registry_auth|length %} +{ +{% for reg in crio_registry_auth %} + "auths": { + "{{ reg.registry }}": { + "auth": "{{ (reg.username + ':' + reg.password) | string | b64encode }}" + } +{% if not loop.last %} + }, +{% else %} + } +{% endif %} +{% endfor %} +} +{% else %} +{} +{% endif %} diff --git a/kubespray/roles/container-engine/cri-o/templates/crio.conf.j2 b/kubespray/roles/container-engine/cri-o/templates/crio.conf.j2 new file mode 100644 index 0000000..f0455d0 --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/templates/crio.conf.j2 @@ -0,0 +1,384 @@ +# The CRI-O configuration file specifies all of the available configuration +# options and command-line flags for the crio(8) OCI Kubernetes Container Runtime +# daemon, but in a TOML format that can be more easily modified and versioned. +# +# Please refer to crio.conf(5) for details of all configuration options. + +# CRI-O supports partial configuration reload during runtime, which can be +# done by sending SIGHUP to the running process. Currently supported options +# are explicitly mentioned with: 'This option supports live configuration +# reload'. + +# CRI-O reads its storage defaults from the containers-storage.conf(5) file +# located at /etc/containers/storage.conf. Modify this storage configuration if +# you want to change the system's defaults. If you want to modify storage just +# for CRI-O, you can change the storage configuration options here. +[crio] + +# Path to the "root directory". CRI-O stores all of its data, including +# containers images, in this directory. +root = "/var/lib/containers/storage" + +# Path to the "run directory". CRI-O stores all of its state in this directory. +# Read from /etc/containers/storage.conf first so unnecessary here +# runroot = "/var/run/containers/storage" + +# Storage driver used to manage the storage of images and containers. Please +# refer to containers-storage.conf(5) to see all available storage drivers. +{% if crio_storage_driver is defined %} +storage_driver = "{{ crio_storage_driver }}" +{% endif %} + +# List to pass options to the storage driver. Please refer to +# containers-storage.conf(5) to see all available storage options. +#storage_option = [ +#] + +# The default log directory where all logs will go unless directly specified by +# the kubelet. The log directory specified must be an absolute directory. +log_dir = "/var/log/crio/pods" + +# Location for CRI-O to lay down the temporary version file. +# It is used to check if crio wipe should wipe containers, which should +# always happen on a node reboot +version_file = "/var/run/crio/version" + +# Location for CRI-O to lay down the persistent version file. +# It is used to check if crio wipe should wipe images, which should +# only happen when CRI-O has been upgraded +version_file_persist = "/var/lib/crio/version" + +# The crio.api table contains settings for the kubelet/gRPC interface. +[crio.api] + +# Path to AF_LOCAL socket on which CRI-O will listen. +listen = "/var/run/crio/crio.sock" + +# IP address on which the stream server will listen. +stream_address = "127.0.0.1" + +# The port on which the stream server will listen. If the port is set to "0", then +# CRI-O will allocate a random free port number. +stream_port = "{{ crio_stream_port }}" + +# Enable encrypted TLS transport of the stream server. +stream_enable_tls = false + +# Path to the x509 certificate file used to serve the encrypted stream. This +# file can change, and CRI-O will automatically pick up the changes within 5 +# minutes. +stream_tls_cert = "" + +# Path to the key file used to serve the encrypted stream. This file can +# change and CRI-O will automatically pick up the changes within 5 minutes. +stream_tls_key = "" + +# Path to the x509 CA(s) file used to verify and authenticate client +# communication with the encrypted stream. This file can change and CRI-O will +# automatically pick up the changes within 5 minutes. +stream_tls_ca = "" + +# Maximum grpc send message size in bytes. If not set or <=0, then CRI-O will default to 16 * 1024 * 1024. +grpc_max_send_msg_size = 16777216 + +# Maximum grpc receive message size. If not set or <= 0, then CRI-O will default to 16 * 1024 * 1024. +grpc_max_recv_msg_size = 16777216 + +# The crio.runtime table contains settings pertaining to the OCI runtime used +# and options for how to set up and manage the OCI runtime. +[crio.runtime] + +# A list of ulimits to be set in containers by default, specified as +# "=:", for example: +# "nofile=1024:2048" +# If nothing is set here, settings will be inherited from the CRI-O daemon +#default_ulimits = [ +#] + +# default_runtime is the _name_ of the OCI runtime to be used as the default. +# The name is matched against the runtimes map below. +default_runtime = "runc" + +# If true, the runtime will not use pivot_root, but instead use MS_MOVE. +no_pivot = false + +# decryption_keys_path is the path where the keys required for +# image decryption are stored. This option supports live configuration reload. +decryption_keys_path = "/etc/crio/keys/" + +# Path to the conmon binary, used for monitoring the OCI runtime. +# Will be searched for using $PATH if empty. +conmon = "{{ crio_conmon }}" + +# Cgroup setting for conmon +{% if crio_cgroup_manager == "cgroupfs" %} +conmon_cgroup = "pod" +{% else %} +{% if kube_reserved is defined and kube_reserved|bool %} +conmon_cgroup = "{{ kube_reserved_cgroups_for_service_slice }}" +{% else %} +conmon_cgroup = "system.slice" +{% endif %} +{% endif %} + +# Environment variable list for the conmon process, used for passing necessary +# environment variables to conmon or the runtime. +conmon_env = [ + "PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin", +] + +# Additional environment variables to set for all the +# containers. These are overridden if set in the +# container image spec or in the container runtime configuration. +default_env = [ +] + +# If true, SELinux will be used for pod separation on the host. +selinux = {{ crio_selinux }} + +# Path to the seccomp.json profile which is used as the default seccomp profile +# for the runtime. If not specified, then the internal default seccomp profile +# will be used. This option supports live configuration reload. +seccomp_profile = "{{ crio_seccomp_profile }}" + +# Used to change the name of the default AppArmor profile of CRI-O. The default +# profile name is "crio-default". This profile only takes effect if the user +# does not specify a profile via the Kubernetes Pod's metadata annotation. If +# the profile is set to "unconfined", then this equals to disabling AppArmor. +# This option supports live configuration reload. +# apparmor_profile = "crio-default" + +# Cgroup management implementation used for the runtime. +cgroup_manager = "{{ crio_cgroup_manager }}" + +# List of default capabilities for containers. If it is empty or commented out, +# only the capabilities defined in the containers json file by the user/kube +# will be added. +default_capabilities = [ + "CHOWN", + "DAC_OVERRIDE", + "FSETID", + "FOWNER", + "NET_RAW", + "SETGID", + "SETUID", + "SETPCAP", + "NET_BIND_SERVICE", + "SYS_CHROOT", + "KILL", +] + +# List of default sysctls. If it is empty or commented out, only the sysctls +# defined in the container json file by the user/kube will be added. +default_sysctls = [ +] + +# List of additional devices. specified as +# "::", for example: "--device=/dev/sdc:/dev/xvdc:rwm". +#If it is empty or commented out, only the devices +# defined in the container json file by the user/kube will be added. +additional_devices = [ +] + +# Path to OCI hooks directories for automatically executed hooks. If one of the +# directories does not exist, then CRI-O will automatically skip them. +hooks_dir = [ + "/usr/share/containers/oci/hooks.d", +] + +# List of default mounts for each container. **Deprecated:** this option will +# be removed in future versions in favor of default_mounts_file. +default_mounts = [ +] + +# Path to the file specifying the defaults mounts for each container. The +# format of the config is /SRC:/DST, one mount per line. Notice that CRI-O reads +# its default mounts from the following two files: +# +# 1) /etc/containers/mounts.conf (i.e., default_mounts_file): This is the +# override file, where users can either add in their own default mounts, or +# override the default mounts shipped with the package. +# +# 2) /usr/share/containers/mounts.conf: This is the default file read for +# mounts. If you want CRI-O to read from a different, specific mounts file, +# you can change the default_mounts_file. Note, if this is done, CRI-O will +# only add mounts it finds in this file. +# +#default_mounts_file = "" + +# Maximum sized allowed for the container log file. Negative numbers indicate +# that no size limit is imposed. If it is positive, it must be >= 8192 to +# match/exceed conmon's read buffer. The file is truncated and re-opened so the +# limit is never exceeded. +log_size_max = -1 + +# Whether container output should be logged to journald in addition to the kuberentes log file +log_to_journald = false + +# Path to directory in which container exit files are written to by conmon. +container_exits_dir = "/var/run/crio/exits" + +# Path to directory for container attach sockets. +container_attach_socket_dir = "/var/run/crio" + +# The prefix to use for the source of the bind mounts. +bind_mount_prefix = "" + +# If set to true, all containers will run in read-only mode. +read_only = false + +# Changes the verbosity of the logs based on the level it is set to. Options +# are fatal, panic, error, warn, info, debug and trace. This option supports +# live configuration reload. +log_level = "{{ crio_log_level }}" + +# Filter the log messages by the provided regular expression. +# This option supports live configuration reload. +log_filter = "" + +# The UID mappings for the user namespace of each container. A range is +# specified in the form containerUID:HostUID:Size. Multiple ranges must be +# separated by comma. +uid_mappings = "" + +# The GID mappings for the user namespace of each container. A range is +# specified in the form containerGID:HostGID:Size. Multiple ranges must be +# separated by comma. +gid_mappings = "" + +# The minimal amount of time in seconds to wait before issuing a timeout +# regarding the proper termination of the container. The lowest possible +# value is 30s, whereas lower values are not considered by CRI-O. +ctr_stop_timeout = 30 + +# **DEPRECATED** this option is being replaced by manage_ns_lifecycle, which is described below. +# manage_network_ns_lifecycle = false + +# manage_ns_lifecycle determines whether we pin and remove namespaces +# and manage their lifecycle +{% if kata_containers_enabled %} +manage_ns_lifecycle = true +{% else %} +manage_ns_lifecycle = false +{% endif %} + +# The directory where the state of the managed namespaces gets tracked. +# Only used when manage_ns_lifecycle is true. +namespaces_dir = "/var/run" + +# pinns_path is the path to find the pinns binary, which is needed to manage namespace lifecycle +{% if bin_dir == "/usr/local/bin" %} +pinns_path = "" +{% else %} +pinns_path = "{{ bin_dir }}/pinns" +{% endif %} + +# The "crio.runtime.runtimes" table defines a list of OCI compatible runtimes. +# The runtime to use is picked based on the runtime_handler provided by the CRI. +# If no runtime_handler is provided, the runtime will be picked based on the level +# of trust of the workload. Each entry in the table should follow the format: +# +#[crio.runtime.runtimes.runtime-handler] +# runtime_path = "/path/to/the/executable" +# runtime_type = "oci" +# runtime_root = "/path/to/the/root" +# +# Where: +# - runtime-handler: name used to identify the runtime +# - runtime_path (optional, string): absolute path to the runtime executable in +# the host filesystem. If omitted, the runtime-handler identifier should match +# the runtime executable name, and the runtime executable should be placed +# in $PATH. +# - runtime_type (optional, string): type of runtime, one of: "oci", "vm". If +# omitted, an "oci" runtime is assumed. +# - runtime_root (optional, string): root directory for storage of containers +# state. + +{% for runtime in crio_runtimes %} +[crio.runtime.runtimes.{{ runtime.name }}] +runtime_path = "{{ runtime.path }}" +runtime_type = "{{ runtime.type }}" +runtime_root = "{{ runtime.root }}" +privileged_without_host_devices = {{ runtime.privileged_without_host_devices|default(false)|lower }} +allowed_annotations = {{ runtime.allowed_annotations|default([])|to_json }} +{% endfor %} + +# Kata Containers with the Firecracker VMM +#[crio.runtime.runtimes.kata-fc] + +# The crio.image table contains settings pertaining to the management of OCI images. +# +# CRI-O reads its configured registries defaults from the system wide +# containers-registries.conf(5) located in /etc/containers/registries.conf. If +# you want to modify just CRI-O, you can change the registries configuration in +# this file. Otherwise, leave insecure_registries and registries commented out to +# use the system's defaults from /etc/containers/registries.conf. +[crio.image] +{% if crio_insecure_registries is defined and crio_insecure_registries|length>0 %} +insecure_registries = {{ crio_insecure_registries }} +{% endif %} + +# Default transport for pulling images from a remote container storage. +default_transport = "docker://" + +# The path to a file containing credentials necessary for pulling images from +# secure registries. The file is similar to that of /var/lib/kubelet/config.json +global_auth_file = "/etc/crio/config.json" + +# The image used to instantiate infra containers. +# This option supports live configuration reload. +pause_image = "{{ crio_pause_image }}" + +# The path to a file containing credentials specific for pulling the pause_image from +# above. The file is similar to that of /var/lib/kubelet/config.json +# This option supports live configuration reload. +pause_image_auth_file = "" + +# The command to run to have a container stay in the paused state. +# When explicitly set to "", it will fallback to the entrypoint and command +# specified in the pause image. When commented out, it will fallback to the +# default: "/pause". This option supports live configuration reload. +pause_command = "/pause" + +# Path to the file which decides what sort of policy we use when deciding +# whether or not to trust an image that we've pulled. It is not recommended that +# this option be used, as the default behavior of using the system-wide default +# policy (i.e., /etc/containers/policy.json) is most often preferred. Please +# refer to containers-policy.json(5) for more details. +signature_policy = "{{ crio_signature_policy }}" + +# Controls how image volumes are handled. The valid values are mkdir, bind and +# ignore; the latter will ignore volumes entirely. +image_volumes = "mkdir" + +# The crio.network table containers settings pertaining to the management of +# CNI plugins. +[crio.network] + +# The default CNI network name to be selected. If not set or "", then +# CRI-O will pick-up the first one found in network_dir. +# cni_default_network = "" + +# Path to the directory where CNI configuration files are located. +network_dir = "/etc/cni/net.d/" + +# Paths to directories where CNI plugin binaries are located. +plugin_dirs = [ + "/opt/cni/bin", + "/usr/libexec/cni", +] + +# A necessary configuration for Prometheus based metrics retrieval +[crio.metrics] + +# Globally enable or disable metrics support. +enable_metrics = {{ crio_enable_metrics | bool | lower }} + +# The port on which the metrics server will listen. +metrics_port = {{ crio_metrics_port }} + +{% if nri_enabled and crio_version >= v1.26.0 %} +[crio.nri] + +enable_nri=true +{% endif %} diff --git a/kubespray/roles/container-engine/cri-o/templates/http-proxy.conf.j2 b/kubespray/roles/container-engine/cri-o/templates/http-proxy.conf.j2 new file mode 100644 index 0000000..212f30f --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/templates/http-proxy.conf.j2 @@ -0,0 +1,2 @@ +[Service] +Environment={% if http_proxy is defined %}"HTTP_PROXY={{ http_proxy }}"{% endif %} {% if https_proxy is defined %}"HTTPS_PROXY={{ https_proxy }}"{% endif %} {% if no_proxy is defined %}"NO_PROXY={{ no_proxy }}"{% endif %} diff --git a/kubespray/roles/container-engine/cri-o/templates/registry.conf.j2 b/kubespray/roles/container-engine/cri-o/templates/registry.conf.j2 new file mode 100644 index 0000000..38368f9 --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/templates/registry.conf.j2 @@ -0,0 +1,13 @@ +[[registry]] +prefix = "{{ item.prefix | default(item.location) }}" +insecure = {{ item.insecure | default('false') | string | lower }} +blocked = {{ item.blocked | default('false') | string | lower }} +location = "{{ item.location }}" +{% if item.mirrors is defined %} +{% for mirror in item.mirrors %} + +[[registry.mirror]] +location = "{{ mirror.location }}" +insecure = {{ mirror.insecure | default('false') | string | lower }} +{% endfor %} +{% endif %} diff --git a/kubespray/roles/container-engine/cri-o/templates/unqualified.conf.j2 b/kubespray/roles/container-engine/cri-o/templates/unqualified.conf.j2 new file mode 100644 index 0000000..fc91f8b --- /dev/null +++ b/kubespray/roles/container-engine/cri-o/templates/unqualified.conf.j2 @@ -0,0 +1,10 @@ +{%- set _unqualified_registries = [] -%} +{% for _registry in crio_registries if _registry.unqualified -%} +{% if _registry.prefix is defined -%} +{{ _unqualified_registries.append(_registry.prefix) }} +{% else %} +{{ _unqualified_registries.append(_registry.location) }} +{%- endif %} +{%- endfor %} + +unqualified-search-registries = {{ _unqualified_registries | string }} diff --git a/kubespray/roles/container-engine/crictl/handlers/main.yml b/kubespray/roles/container-engine/crictl/handlers/main.yml new file mode 100644 index 0000000..5319586 --- /dev/null +++ b/kubespray/roles/container-engine/crictl/handlers/main.yml @@ -0,0 +1,12 @@ +--- +- name: Get crictl completion + command: "{{ bin_dir }}/crictl completion" + changed_when: False + register: cri_completion + check_mode: false + +- name: Install crictl completion + copy: + dest: /etc/bash_completion.d/crictl + content: "{{ cri_completion.stdout }}" + mode: 0644 diff --git a/kubespray/roles/container-engine/crictl/tasks/crictl.yml b/kubespray/roles/container-engine/crictl/tasks/crictl.yml new file mode 100644 index 0000000..cffa050 --- /dev/null +++ b/kubespray/roles/container-engine/crictl/tasks/crictl.yml @@ -0,0 +1,22 @@ +--- +- name: Crictl | Download crictl + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.crictl) }}" + +- name: Install crictl config + template: + src: crictl.yaml.j2 + dest: /etc/crictl.yaml + owner: root + mode: 0644 + +- name: Copy crictl binary from download dir + copy: + src: "{{ local_release_dir }}/crictl" + dest: "{{ bin_dir }}/crictl" + mode: 0755 + remote_src: true + notify: + - Get crictl completion + - Install crictl completion diff --git a/kubespray/roles/container-engine/crictl/tasks/main.yml b/kubespray/roles/container-engine/crictl/tasks/main.yml new file mode 100644 index 0000000..9337016 --- /dev/null +++ b/kubespray/roles/container-engine/crictl/tasks/main.yml @@ -0,0 +1,3 @@ +--- +- name: Install crictl + include_tasks: crictl.yml diff --git a/kubespray/roles/container-engine/crictl/templates/crictl.yaml.j2 b/kubespray/roles/container-engine/crictl/templates/crictl.yaml.j2 new file mode 100644 index 0000000..b97dbef --- /dev/null +++ b/kubespray/roles/container-engine/crictl/templates/crictl.yaml.j2 @@ -0,0 +1,4 @@ +runtime-endpoint: {{ cri_socket }} +image-endpoint: {{ cri_socket }} +timeout: 30 +debug: false diff --git a/kubespray/roles/container-engine/crun/tasks/main.yml b/kubespray/roles/container-engine/crun/tasks/main.yml new file mode 100644 index 0000000..c21bb3f --- /dev/null +++ b/kubespray/roles/container-engine/crun/tasks/main.yml @@ -0,0 +1,12 @@ +--- +- name: Crun | Download crun binary + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.crun) }}" + +- name: Copy crun binary from download dir + copy: + src: "{{ downloads.crun.dest }}" + dest: "{{ bin_dir }}/crun" + mode: 0755 + remote_src: true diff --git a/kubespray/roles/container-engine/docker-storage/defaults/main.yml b/kubespray/roles/container-engine/docker-storage/defaults/main.yml new file mode 100644 index 0000000..6a69556 --- /dev/null +++ b/kubespray/roles/container-engine/docker-storage/defaults/main.yml @@ -0,0 +1,19 @@ +--- +docker_container_storage_setup_repository: https://github.com/projectatomic/container-storage-setup.git +docker_container_storage_setup_version: v0.6.0 +docker_container_storage_setup_profile_name: kubespray +docker_container_storage_setup_storage_driver: devicemapper +docker_container_storage_setup_container_thinpool: docker-pool +# It must be define a disk path for docker_container_storage_setup_devs. +# Otherwise docker-storage-setup will be executed incorrectly. +# docker_container_storage_setup_devs: /dev/vdb +docker_container_storage_setup_data_size: 40%FREE +docker_container_storage_setup_min_data_size: 2G +docker_container_storage_setup_chunk_size: 512K +docker_container_storage_setup_growpart: "false" +docker_container_storage_setup_auto_extend_pool: "yes" +docker_container_storage_setup_pool_autoextend_threshold: 60 +docker_container_storage_setup_pool_autoextend_percent: 20 +docker_container_storage_setup_device_wait_timeout: 60 +docker_container_storage_setup_wipe_signatures: "false" +docker_container_storage_setup_container_root_lv_size: 40%FREE diff --git a/kubespray/roles/container-engine/docker-storage/files/install_container_storage_setup.sh b/kubespray/roles/container-engine/docker-storage/files/install_container_storage_setup.sh new file mode 100644 index 0000000..604c843 --- /dev/null +++ b/kubespray/roles/container-engine/docker-storage/files/install_container_storage_setup.sh @@ -0,0 +1,23 @@ +#!/bin/sh + +set -e + +repository=${1:-https://github.com/projectatomic/container-storage-setup.git} +version=${2:-master} +profile_name=${3:-kubespray} +dir=`mktemp -d` +export GIT_DIR=$dir/.git +export GIT_WORK_TREE=$dir + +git init +git fetch --depth 1 $repository $version +git merge FETCH_HEAD +make -C $dir install +rm -rf /var/lib/container-storage-setup/$profile_name $dir + +set +e + +/usr/bin/container-storage-setup create $profile_name /etc/sysconfig/docker-storage-setup && /usr/bin/container-storage-setup activate $profile_name +# FIXME: exit status can be 1 for both fatal and non fatal errors in current release, +# could be improved by matching error strings +exit 0 diff --git a/kubespray/roles/container-engine/docker-storage/tasks/main.yml b/kubespray/roles/container-engine/docker-storage/tasks/main.yml new file mode 100644 index 0000000..ec12975 --- /dev/null +++ b/kubespray/roles/container-engine/docker-storage/tasks/main.yml @@ -0,0 +1,48 @@ +--- + +- name: Docker-storage-setup | install git and make + with_items: [git, make] + package: + pkg: "{{ item }}" + state: present + +- name: Docker-storage-setup | docker-storage-setup sysconfig template + template: + src: docker-storage-setup.j2 + dest: /etc/sysconfig/docker-storage-setup + mode: 0644 + +- name: Docker-storage-override-directory | docker service storage-setup override dir + file: + dest: /etc/systemd/system/docker.service.d + mode: 0755 + owner: root + group: root + state: directory + +- name: Docker-storage-override | docker service storage-setup override file + copy: + dest: /etc/systemd/system/docker.service.d/override.conf + content: |- + ### This file is managed by Ansible + [Service] + EnvironmentFile=-/etc/sysconfig/docker-storage + + owner: root + group: root + mode: 0644 + +# https://docs.docker.com/engine/installation/linux/docker-ce/centos/#install-using-the-repository +- name: Docker-storage-setup | install lvm2 + package: + name: lvm2 + state: present + +- name: Docker-storage-setup | install and run container-storage-setup + become: yes + script: | + install_container_storage_setup.sh \ + {{ docker_container_storage_setup_repository }} \ + {{ docker_container_storage_setup_version }} \ + {{ docker_container_storage_setup_profile_name }} + notify: Docker | reload systemd diff --git a/kubespray/roles/container-engine/docker-storage/templates/docker-storage-setup.j2 b/kubespray/roles/container-engine/docker-storage/templates/docker-storage-setup.j2 new file mode 100644 index 0000000..1a502b2 --- /dev/null +++ b/kubespray/roles/container-engine/docker-storage/templates/docker-storage-setup.j2 @@ -0,0 +1,35 @@ +{%if docker_container_storage_setup_storage_driver is defined%}STORAGE_DRIVER={{docker_container_storage_setup_storage_driver}}{%endif%} + +{%if docker_container_storage_setup_extra_storage_options is defined%}EXTRA_STORAGE_OPTIONS={{docker_container_storage_setup_extra_storage_options}}{%endif%} + +{%if docker_container_storage_setup_devs is defined%}DEVS={{docker_container_storage_setup_devs}}{%endif%} + +{%if docker_container_storage_setup_container_thinpool is defined%}CONTAINER_THINPOOL={{docker_container_storage_setup_container_thinpool}}{%endif%} + +{%if docker_container_storage_setup_vg is defined%}VG={{docker_container_storage_setup_vg}}{%endif%} + +{%if docker_container_storage_setup_root_size is defined%}ROOT_SIZE={{docker_container_storage_setup_root_size}}{%endif%} + +{%if docker_container_storage_setup_data_size is defined%}DATA_SIZE={{docker_container_storage_setup_data_size}}{%endif%} + +{%if docker_container_storage_setup_min_data_size is defined%}MIN_DATA_SIZE={{docker_container_storage_setup_min_data_size}}{%endif%} + +{%if docker_container_storage_setup_chunk_size is defined%}CHUNK_SIZE={{docker_container_storage_setup_chunk_size}}{%endif%} + +{%if docker_container_storage_setup_growpart is defined%}GROWPART={{docker_container_storage_setup_growpart}}{%endif%} + +{%if docker_container_storage_setup_auto_extend_pool is defined%}AUTO_EXTEND_POOL={{docker_container_storage_setup_auto_extend_pool}}{%endif%} + +{%if docker_container_storage_setup_pool_autoextend_threshold is defined%}POOL_AUTOEXTEND_THRESHOLD={{docker_container_storage_setup_pool_autoextend_threshold}}{%endif%} + +{%if docker_container_storage_setup_pool_autoextend_percent is defined%}POOL_AUTOEXTEND_PERCENT={{docker_container_storage_setup_pool_autoextend_percent}}{%endif%} + +{%if docker_container_storage_setup_device_wait_timeout is defined%}DEVICE_WAIT_TIMEOUT={{docker_container_storage_setup_device_wait_timeout}}{%endif%} + +{%if docker_container_storage_setup_wipe_signatures is defined%}WIPE_SIGNATURES={{docker_container_storage_setup_wipe_signatures}}{%endif%} + +{%if docker_container_storage_setup_container_root_lv_name is defined%}CONTAINER_ROOT_LV_NAME={{docker_container_storage_setup_container_root_lv_name}}{%endif%} + +{%if docker_container_storage_setup_container_root_lv_size is defined%}CONTAINER_ROOT_LV_SIZE={{docker_container_storage_setup_container_root_lv_size}}{%endif%} + +{%if docker_container_storage_setup_container_root_lv_mount_path is defined%}CONTAINER_ROOT_LV_MOUNT_PATH={{docker_container_storage_setup_container_root_lv_mount_path}}{%endif%} diff --git a/kubespray/roles/container-engine/docker/defaults/main.yml b/kubespray/roles/container-engine/docker/defaults/main.yml new file mode 100644 index 0000000..91227f9 --- /dev/null +++ b/kubespray/roles/container-engine/docker/defaults/main.yml @@ -0,0 +1,64 @@ +--- +docker_version: '20.10' +docker_cli_version: "{{ docker_version }}" + +docker_package_info: + pkgs: + +docker_repo_key_info: + repo_keys: + +docker_repo_info: + repos: + +docker_cgroup_driver: systemd + +docker_bin_dir: "/usr/bin" + +# flag to enable/disable docker cleanup +docker_orphan_clean_up: false + +# old docker package names to be removed +docker_remove_packages_yum: + - docker + - docker-common + - docker-engine + - docker-selinux.noarch + - docker-client + - docker-client-latest + - docker-latest + - docker-latest-logrotate + - docker-logrotate + - docker-engine-selinux.noarch + +# remove podman to avoid containerd.io confliction +podman_remove_packages_yum: + - podman + +docker_remove_packages_apt: + - docker + - docker-engine + - docker.io + +# Docker specific repos should be part of the docker role not containerd-common anymore +# Optional values for containerd apt repo +containerd_package_info: + pkgs: + +# Fedora docker-ce repo +docker_fedora_repo_base_url: 'https://download.docker.com/linux/fedora/{{ ansible_distribution_major_version }}/$basearch/stable' +docker_fedora_repo_gpgkey: 'https://download.docker.com/linux/fedora/gpg' + +# CentOS/RedHat docker-ce repo +docker_rh_repo_base_url: 'https://download.docker.com/linux/centos/{{ ansible_distribution_major_version }}/$basearch/stable' +docker_rh_repo_gpgkey: 'https://download.docker.com/linux/centos/gpg' + +# Ubuntu docker-ce repo +docker_ubuntu_repo_base_url: "https://download.docker.com/linux/ubuntu" +docker_ubuntu_repo_gpgkey: 'https://download.docker.com/linux/ubuntu/gpg' +docker_ubuntu_repo_repokey: '9DC858229FC7DD38854AE2D88D81803C0EBFCD88' + +# Debian docker-ce repo +docker_debian_repo_base_url: "https://download.docker.com/linux/debian" +docker_debian_repo_gpgkey: 'https://download.docker.com/linux/debian/gpg' +docker_debian_repo_repokey: '9DC858229FC7DD38854AE2D88D81803C0EBFCD88' diff --git a/kubespray/roles/container-engine/docker/files/cleanup-docker-orphans.sh b/kubespray/roles/container-engine/docker/files/cleanup-docker-orphans.sh new file mode 100644 index 0000000..d7a9a8f --- /dev/null +++ b/kubespray/roles/container-engine/docker/files/cleanup-docker-orphans.sh @@ -0,0 +1,38 @@ +#!/bin/bash +list_descendants () +{ + local children=$(ps -o pid= --ppid "$1") + for pid in $children + do + list_descendants "$pid" + done + [[ -n "$children" ]] && echo "$children" +} + +shim_search="^docker-containerd-shim|^containerd-shim" +count_shim_processes=$(pgrep -f $shim_search | wc -l) + +if [ ${count_shim_processes} -gt 0 ]; then + # Find all container pids from shims + orphans=$(pgrep -P $(pgrep -d ',' -f $shim_search) |\ + # Filter out valid docker pids, leaving the orphans + egrep -v $(docker ps -q | xargs docker inspect --format '{{.State.Pid}}' | awk '{printf "%s%s",sep,$1; sep="|"}')) + + if [[ -n "$orphans" && -n "$(ps -o ppid= $orphans)" ]] + then + # Get shim pids of orphans + orphan_shim_pids=$(ps -o pid= $(ps -o ppid= $orphans)) + + # Find all orphaned container PIDs + orphan_container_pids=$(for pid in $orphan_shim_pids; do list_descendants $pid; done) + + # Recursively kill all child PIDs of orphan shims + echo -e "Killing orphan container PIDs and descendants: \n$(ps -O ppid= $orphan_container_pids)" + kill -9 $orphan_container_pids || true + + else + echo "No orphaned containers found" + fi +else + echo "The node doesn't have any shim processes." +fi diff --git a/kubespray/roles/container-engine/docker/handlers/main.yml b/kubespray/roles/container-engine/docker/handlers/main.yml new file mode 100644 index 0000000..14a7b39 --- /dev/null +++ b/kubespray/roles/container-engine/docker/handlers/main.yml @@ -0,0 +1,32 @@ +--- +- name: Restart docker + command: /bin/true + notify: + - Docker | reload systemd + - Docker | reload docker.socket + - Docker | reload docker + - Docker | wait for docker + +- name: Docker | reload systemd + systemd: + name: docker + daemon_reload: true + masked: no + +- name: Docker | reload docker.socket + service: + name: docker.socket + state: restarted + when: ansible_os_family in ['Flatcar', 'Flatcar Container Linux by Kinvolk'] or is_fedora_coreos + +- name: Docker | reload docker + service: + name: docker + state: restarted + +- name: Docker | wait for docker + command: "{{ docker_bin_dir }}/docker images" + register: docker_ready + retries: 20 + delay: 1 + until: docker_ready.rc == 0 diff --git a/kubespray/roles/container-engine/docker/meta/main.yml b/kubespray/roles/container-engine/docker/meta/main.yml new file mode 100644 index 0000000..d7e4751 --- /dev/null +++ b/kubespray/roles/container-engine/docker/meta/main.yml @@ -0,0 +1,5 @@ +--- +dependencies: + - role: container-engine/containerd-common + - role: container-engine/docker-storage + when: docker_container_storage_setup and ansible_os_family == "RedHat" diff --git a/kubespray/roles/container-engine/docker/tasks/docker_plugin.yml b/kubespray/roles/container-engine/docker/tasks/docker_plugin.yml new file mode 100644 index 0000000..8ee530e --- /dev/null +++ b/kubespray/roles/container-engine/docker/tasks/docker_plugin.yml @@ -0,0 +1,8 @@ +--- +- name: Install Docker plugin + command: docker plugin install --grant-all-permissions {{ docker_plugin | quote }} + when: docker_plugin is defined + register: docker_plugin_status + failed_when: + - docker_plugin_status.failed + - '"already exists" not in docker_plugin_status.stderr' diff --git a/kubespray/roles/container-engine/docker/tasks/main.yml b/kubespray/roles/container-engine/docker/tasks/main.yml new file mode 100644 index 0000000..cf81ce2 --- /dev/null +++ b/kubespray/roles/container-engine/docker/tasks/main.yml @@ -0,0 +1,180 @@ +--- +- name: Check if fedora coreos + stat: + path: /run/ostree-booted + get_attributes: no + get_checksum: no + get_mime: no + register: ostree + +- name: Set is_ostree + set_fact: + is_ostree: "{{ ostree.stat.exists }}" + +- name: Set docker_version for openEuler + set_fact: + docker_version: '19.03' + when: ansible_distribution == "openEuler" + tags: + - facts + +- name: Gather os specific variables + include_vars: "{{ item }}" + with_first_found: + - files: + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_release | lower }}-{{ host_architecture }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_release | lower }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_major_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}-{{ host_architecture }}.yml" + - "{{ ansible_distribution | lower }}.yml" + - "{{ ansible_distribution.split(' ')[0] | lower }}.yml" + - "{{ ansible_os_family | lower }}-{{ ansible_distribution_major_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_os_family | lower }}-{{ host_architecture }}.yml" + - "{{ ansible_os_family | lower }}.yml" + - defaults.yml + paths: + - ../vars + skip: true + tags: + - facts + +- name: Warn about Docker version on SUSE + debug: + msg: "SUSE distributions always install Docker from the distro repos" + when: ansible_pkg_mgr == 'zypper' + +- name: Gather DNS facts + include_tasks: set_facts_dns.yml + when: dns_mode != 'none' and resolvconf_mode == 'docker_dns' + tags: + - facts + +- name: Pre-upgrade docker + import_tasks: pre-upgrade.yml + +- name: Ensure docker-ce repository public key is installed + apt_key: + id: "{{ item }}" + url: "{{ docker_repo_key_info.url }}" + state: present + register: keyserver_task_result + until: keyserver_task_result is succeeded + retries: 4 + delay: "{{ retry_stagger | d(3) }}" + with_items: "{{ docker_repo_key_info.repo_keys }}" + environment: "{{ proxy_env }}" + when: ansible_pkg_mgr == 'apt' + +- name: Ensure docker-ce repository is enabled + apt_repository: + repo: "{{ item }}" + state: present + with_items: "{{ docker_repo_info.repos }}" + when: ansible_pkg_mgr == 'apt' + +- name: Configure docker repository on Fedora + template: + src: "fedora_docker.repo.j2" + dest: "{{ yum_repo_dir }}/docker.repo" + mode: 0644 + when: ansible_distribution == "Fedora" and not is_ostree + +- name: Configure docker repository on RedHat/CentOS/OracleLinux/AlmaLinux/KylinLinux + template: + src: "rh_docker.repo.j2" + dest: "{{ yum_repo_dir }}/docker-ce.repo" + mode: 0644 + when: + - ansible_os_family == "RedHat" + - ansible_distribution != "Fedora" + - not is_ostree + +- name: Remove dpkg hold + dpkg_selections: + name: "{{ item }}" + selection: install + when: ansible_pkg_mgr == 'apt' + changed_when: false + with_items: + - "{{ containerd_package }}" + - docker-ce + - docker-ce-cli + +- name: Ensure docker packages are installed + package: + name: "{{ docker_package_info.pkgs }}" + state: "{{ docker_package_info.state | default('present') }}" + module_defaults: + apt: + update_cache: true + dnf: + enablerepo: "{{ docker_package_info.enablerepo | default(omit) }}" + disablerepo: "{{ docker_package_info.disablerepo | default(omit) }}" + yum: + enablerepo: "{{ docker_package_info.enablerepo | default(omit) }}" + zypper: + update_cache: true + register: docker_task_result + until: docker_task_result is succeeded + retries: 4 + delay: "{{ retry_stagger | d(3) }}" + notify: Restart docker + when: + - not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + - not is_ostree + - docker_package_info.pkgs | length > 0 + +# This is required to ensure any apt upgrade will not break kubernetes +- name: Tell Debian hosts not to change the docker version with apt upgrade + dpkg_selections: + name: "{{ item }}" + selection: hold + when: ansible_pkg_mgr == 'apt' + changed_when: false + with_items: + - "{{ containerd_package }}" + - docker-ce + - docker-ce-cli + +- name: Ensure docker started, remove our config if docker start failed and try again + block: + - name: Ensure service is started if docker packages are already present + service: + name: docker + state: started + when: docker_task_result is not changed + rescue: + - debug: # noqa name[missing] + msg: "Docker start failed. Try to remove our config" + - name: Remove kubespray generated config + file: + path: "{{ item }}" + state: absent + with_items: + - /etc/systemd/system/docker.service.d/http-proxy.conf + - /etc/systemd/system/docker.service.d/docker-options.conf + - /etc/systemd/system/docker.service.d/docker-dns.conf + - /etc/systemd/system/docker.service.d/docker-orphan-cleanup.conf + notify: Restart docker + +- name: Flush handlers so we can wait for docker to come up + meta: flush_handlers + +# Install each plugin using a looped include to make error handling in the included task simpler. +- name: Install docker plugin + include_tasks: docker_plugin.yml + loop: "{{ docker_plugins }}" + loop_control: + loop_var: docker_plugin + +- name: Set docker systemd config + import_tasks: systemd.yml + +- name: Ensure docker service is started and enabled + service: + name: "{{ item }}" + enabled: yes + state: started + with_items: + - docker diff --git a/kubespray/roles/container-engine/docker/tasks/pre-upgrade.yml b/kubespray/roles/container-engine/docker/tasks/pre-upgrade.yml new file mode 100644 index 0000000..f346b46 --- /dev/null +++ b/kubespray/roles/container-engine/docker/tasks/pre-upgrade.yml @@ -0,0 +1,36 @@ +--- +- name: Remove legacy docker repo file + file: + path: "{{ yum_repo_dir }}/docker.repo" + state: absent + when: + - ansible_os_family == 'RedHat' + - not is_ostree + +- name: Ensure old versions of Docker are not installed. | Debian + apt: + name: '{{ docker_remove_packages_apt }}' + state: absent + when: + - ansible_os_family == 'Debian' + - (docker_versioned_pkg[docker_version | string] is search('docker-ce')) + + +- name: Ensure podman not installed. | RedHat + package: + name: '{{ podman_remove_packages_yum }}' + state: absent + when: + - ansible_os_family == 'RedHat' + - (docker_versioned_pkg[docker_version | string] is search('docker-ce')) + - not is_ostree + + +- name: Ensure old versions of Docker are not installed. | RedHat + package: + name: '{{ docker_remove_packages_yum }}' + state: absent + when: + - ansible_os_family == 'RedHat' + - (docker_versioned_pkg[docker_version | string] is search('docker-ce')) + - not is_ostree diff --git a/kubespray/roles/container-engine/docker/tasks/reset.yml b/kubespray/roles/container-engine/docker/tasks/reset.yml new file mode 100644 index 0000000..4bca908 --- /dev/null +++ b/kubespray/roles/container-engine/docker/tasks/reset.yml @@ -0,0 +1,106 @@ +--- + +- name: Docker | Get package facts + package_facts: + manager: auto + +- name: Docker | Find docker packages + set_fact: + docker_packages_list: "{{ ansible_facts.packages.keys() | select('search', '^docker*') }}" + containerd_package: "{{ ansible_facts.packages.keys() | select('search', '^containerd*') }}" + +- name: Docker | Stop all running container + shell: "set -o pipefail && {{ docker_bin_dir }}/docker ps -q | xargs -r {{ docker_bin_dir }}/docker kill" + args: + executable: /bin/bash + register: stop_all_containers + retries: 5 + until: stop_all_containers.rc == 0 + changed_when: true + delay: 5 + ignore_errors: true # noqa ignore-errors + when: docker_packages_list | length>0 + +- name: Reset | remove all containers + shell: "set -o pipefail && {{ docker_bin_dir }}/docker ps -aq | xargs -r docker rm -fv" + args: + executable: /bin/bash + register: remove_all_containers + retries: 4 + until: remove_all_containers.rc == 0 + delay: 5 + when: docker_packages_list | length>0 + +- name: Docker | Stop docker service + service: + name: "{{ item }}" + enabled: false + state: stopped + loop: + - docker + - docker.socket + - containerd + when: docker_packages_list | length>0 + +- name: Docker | Remove dpkg hold + dpkg_selections: + name: "{{ item }}" + selection: install + when: ansible_pkg_mgr == 'apt' + changed_when: false + with_items: + - "{{ docker_packages_list }}" + - "{{ containerd_package }}" + +- name: Docker | Remove docker package + package: + name: "{{ item }}" + state: absent + changed_when: false + with_items: + - "{{ docker_packages_list }}" + - "{{ containerd_package }}" + when: + - not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + - not is_ostree + - docker_packages_list | length > 0 + +- name: Docker | ensure docker-ce repository is removed + apt_repository: + repo: "{{ item }}" + state: absent + with_items: "{{ docker_repo_info.repos }}" + when: ansible_pkg_mgr == 'apt' + +- name: Docker | Remove docker repository on Fedora + file: + name: "{{ yum_repo_dir }}/docker.repo" + state: absent + when: ansible_distribution == "Fedora" and not is_ostree + +- name: Docker | Remove docker repository on RedHat/CentOS/Oracle/AlmaLinux Linux + file: + name: "{{ yum_repo_dir }}/docker-ce.repo" + state: absent + when: + - ansible_os_family == "RedHat" + - ansible_distribution != "Fedora" + - not is_ostree + +- name: Docker | Remove docker configuration files + file: + name: "{{ item }}" + state: absent + loop: + - /etc/systemd/system/docker.service.d/ + - /etc/systemd/system/docker.socket + - /etc/systemd/system/docker.service + - /etc/systemd/system/containerd.service + - /etc/systemd/system/containerd.service.d + - /var/lib/docker + - /etc/docker + ignore_errors: true # noqa ignore-errors + +- name: Docker | systemctl daemon-reload # noqa no-handler + systemd: + daemon_reload: true diff --git a/kubespray/roles/container-engine/docker/tasks/set_facts_dns.yml b/kubespray/roles/container-engine/docker/tasks/set_facts_dns.yml new file mode 100644 index 0000000..d7c1039 --- /dev/null +++ b/kubespray/roles/container-engine/docker/tasks/set_facts_dns.yml @@ -0,0 +1,66 @@ +--- + +- name: Set dns server for docker + set_fact: + docker_dns_servers: "{{ dns_servers }}" + +- name: Show docker_dns_servers + debug: + msg: "{{ docker_dns_servers }}" + +- name: Add upstream dns servers + set_fact: + docker_dns_servers: "{{ docker_dns_servers + upstream_dns_servers | default([]) }}" + when: dns_mode in ['coredns', 'coredns_dual'] + +- name: Add global searchdomains + set_fact: + docker_dns_search_domains: "{{ docker_dns_search_domains + searchdomains | default([]) }}" + +- name: Check system nameservers + shell: set -o pipefail && grep "^nameserver" /etc/resolv.conf | sed -r 's/^nameserver\s*([^#\s]+)\s*(#.*)?/\1/' + args: + executable: /bin/bash + changed_when: False + register: system_nameservers + check_mode: no + +- name: Check system search domains + # noqa risky-shell-pipe - if resolf.conf has no search domain, grep will exit 1 which would force us to add failed_when: false + # Therefore -o pipefail is not applicable in this specific instance + shell: grep "^search" /etc/resolv.conf | sed -r 's/^search\s*([^#]+)\s*(#.*)?/\1/' + args: + executable: /bin/bash + changed_when: False + register: system_search_domains + check_mode: no + +- name: Add system nameservers to docker options + set_fact: + docker_dns_servers: "{{ docker_dns_servers | union(system_nameservers.stdout_lines) | unique }}" + when: system_nameservers.stdout + +- name: Add system search domains to docker options + set_fact: + docker_dns_search_domains: "{{ docker_dns_search_domains | union(system_search_domains.stdout.split() | default([])) | unique }}" + when: system_search_domains.stdout + +- name: Check number of nameservers + fail: + msg: "Too many nameservers. You can relax this check by set docker_dns_servers_strict=false in docker.yml and we will only use the first 3." + when: docker_dns_servers | length > 3 and docker_dns_servers_strict | bool + +- name: Rtrim number of nameservers to 3 + set_fact: + docker_dns_servers: "{{ docker_dns_servers[0:3] }}" + when: docker_dns_servers | length > 3 and not docker_dns_servers_strict | bool + +- name: Check number of search domains + fail: + msg: "Too many search domains" + when: docker_dns_search_domains | length > 6 + +- name: Check length of search domains + fail: + msg: "Search domains exceeded limit of 256 characters" + when: docker_dns_search_domains | join(' ') | length > 256 diff --git a/kubespray/roles/container-engine/docker/tasks/systemd.yml b/kubespray/roles/container-engine/docker/tasks/systemd.yml new file mode 100644 index 0000000..57d9b9c --- /dev/null +++ b/kubespray/roles/container-engine/docker/tasks/systemd.yml @@ -0,0 +1,68 @@ +--- +- name: Create docker service systemd directory if it doesn't exist + file: + path: /etc/systemd/system/docker.service.d + state: directory + mode: 0755 + +- name: Write docker proxy drop-in + template: + src: http-proxy.conf.j2 + dest: /etc/systemd/system/docker.service.d/http-proxy.conf + mode: 0644 + notify: Restart docker + when: http_proxy is defined or https_proxy is defined + +- name: Get systemd version + # noqa command-instead-of-module - systemctl is called intentionally here + shell: set -o pipefail && systemctl --version | head -n 1 | cut -d " " -f 2 + args: + executable: /bin/bash + register: systemd_version + when: not is_ostree + changed_when: false + check_mode: false + +- name: Write docker.service systemd file + template: + src: docker.service.j2 + dest: /etc/systemd/system/docker.service + mode: 0644 + register: docker_service_file + notify: Restart docker + when: + - not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + - not is_fedora_coreos + +- name: Write docker options systemd drop-in + template: + src: docker-options.conf.j2 + dest: "/etc/systemd/system/docker.service.d/docker-options.conf" + mode: 0644 + notify: Restart docker + +- name: Write docker dns systemd drop-in + template: + src: docker-dns.conf.j2 + dest: "/etc/systemd/system/docker.service.d/docker-dns.conf" + mode: 0644 + notify: Restart docker + when: dns_mode != 'none' and resolvconf_mode == 'docker_dns' + +- name: Copy docker orphan clean up script to the node + copy: + src: cleanup-docker-orphans.sh + dest: "{{ bin_dir }}/cleanup-docker-orphans.sh" + mode: 0755 + when: docker_orphan_clean_up | bool + +- name: Write docker orphan clean up systemd drop-in + template: + src: docker-orphan-cleanup.conf.j2 + dest: "/etc/systemd/system/docker.service.d/docker-orphan-cleanup.conf" + mode: 0644 + notify: Restart docker + when: docker_orphan_clean_up | bool + +- name: Flush handlers + meta: flush_handlers diff --git a/kubespray/roles/container-engine/docker/templates/docker-dns.conf.j2 b/kubespray/roles/container-engine/docker/templates/docker-dns.conf.j2 new file mode 100644 index 0000000..01dbd3b --- /dev/null +++ b/kubespray/roles/container-engine/docker/templates/docker-dns.conf.j2 @@ -0,0 +1,6 @@ +[Service] +Environment="DOCKER_DNS_OPTIONS=\ + {% for d in docker_dns_servers %}--dns {{ d }} {% endfor %} \ + {% for d in docker_dns_search_domains %}--dns-search {{ d }} {% endfor %} \ + {% for o in docker_dns_options %}--dns-opt {{ o }} {% endfor %} \ +" diff --git a/kubespray/roles/container-engine/docker/templates/docker-options.conf.j2 b/kubespray/roles/container-engine/docker/templates/docker-options.conf.j2 new file mode 100644 index 0000000..ae661ad --- /dev/null +++ b/kubespray/roles/container-engine/docker/templates/docker-options.conf.j2 @@ -0,0 +1,11 @@ +[Service] +Environment="DOCKER_OPTS={{ docker_options|default('') }} --iptables={{ docker_iptables_enabled | default('false') }} \ +--exec-opt native.cgroupdriver={{ docker_cgroup_driver }} \ +{% for i in docker_insecure_registries %}--insecure-registry={{ i }} {% endfor %} \ +{% for i in docker_registry_mirrors %}--registry-mirror={{ i }} {% endfor %} \ +--data-root={{ docker_daemon_graph }} \ +{% if ansible_os_family not in ["openSUSE Leap", "openSUSE Tumbleweed", "Suse"] %}{{ docker_log_opts }}{% endif %}" + +{% if docker_mount_flags is defined and docker_mount_flags != "" %} +MountFlags={{ docker_mount_flags }} +{% endif %} diff --git a/kubespray/roles/container-engine/docker/templates/docker-orphan-cleanup.conf.j2 b/kubespray/roles/container-engine/docker/templates/docker-orphan-cleanup.conf.j2 new file mode 100644 index 0000000..370f1e7 --- /dev/null +++ b/kubespray/roles/container-engine/docker/templates/docker-orphan-cleanup.conf.j2 @@ -0,0 +1,2 @@ +[Service] +ExecStartPost=-{{ bin_dir }}/cleanup-docker-orphans.sh diff --git a/kubespray/roles/container-engine/docker/templates/docker.service.j2 b/kubespray/roles/container-engine/docker/templates/docker.service.j2 new file mode 100644 index 0000000..539c3a5 --- /dev/null +++ b/kubespray/roles/container-engine/docker/templates/docker.service.j2 @@ -0,0 +1,51 @@ +[Unit] +Description=Docker Application Container Engine +Documentation=http://docs.docker.com +{% if ansible_os_family == "RedHat" %} +After=network.target {{ ' docker-storage-setup.service' if docker_container_storage_setup else '' }} containerd.service +BindsTo=containerd.service +{{ 'Wants=docker-storage-setup.service' if docker_container_storage_setup else '' }} +{% elif ansible_os_family == "Debian" %} +After=network.target docker.socket containerd.service +BindsTo=containerd.service +Wants=docker.socket +{% elif ansible_os_family == "Suse" %} +After=network.target lvm2-monitor.service SuSEfirewall2.service +# After=network.target containerd.service +# BindsTo=containerd.service +{% endif %} + +[Service] +Type=notify +{% if docker_storage_options is defined %} +Environment="DOCKER_STORAGE_OPTIONS={{ docker_storage_options }}" +{% endif %} +Environment=GOTRACEBACK=crash +ExecReload=/bin/kill -s HUP $MAINPID +Delegate=yes +KillMode=process +ExecStart={{ docker_bin_dir }}/dockerd \ +{% if ansible_os_family == "Suse" %} + --add-runtime oci=/usr/sbin/docker-runc \ +{% endif %} + $DOCKER_OPTS \ + $DOCKER_STORAGE_OPTIONS \ + $DOCKER_DNS_OPTIONS +{% if not is_ostree and systemd_version.stdout|int >= 226 %} +TasksMax=infinity +{% endif %} +LimitNOFILE=1048576 +LimitNPROC=1048576 +LimitCORE=infinity +TimeoutStartSec=1min +# restart the docker process if it exits prematurely +Restart=on-failure +StartLimitBurst=3 +StartLimitInterval=60s +# Set the cgroup slice of the service so that kube reserved takes effect +{% if kube_reserved is defined and kube_reserved|bool %} +Slice={{ kube_reserved_cgroups_for_service_slice }} +{% endif %} + +[Install] +WantedBy=multi-user.target diff --git a/kubespray/roles/container-engine/docker/templates/fedora_docker.repo.j2 b/kubespray/roles/container-engine/docker/templates/fedora_docker.repo.j2 new file mode 100644 index 0000000..3958ff0 --- /dev/null +++ b/kubespray/roles/container-engine/docker/templates/fedora_docker.repo.j2 @@ -0,0 +1,7 @@ +[docker-ce] +name=Docker-CE Repository +baseurl={{ docker_fedora_repo_base_url }} +enabled=1 +gpgcheck={{ '1' if docker_fedora_repo_gpgkey else '0' }} +gpgkey={{ docker_fedora_repo_gpgkey }} +{% if http_proxy is defined %}proxy={{ http_proxy }}{% endif %} diff --git a/kubespray/roles/container-engine/docker/templates/http-proxy.conf.j2 b/kubespray/roles/container-engine/docker/templates/http-proxy.conf.j2 new file mode 100644 index 0000000..212f30f --- /dev/null +++ b/kubespray/roles/container-engine/docker/templates/http-proxy.conf.j2 @@ -0,0 +1,2 @@ +[Service] +Environment={% if http_proxy is defined %}"HTTP_PROXY={{ http_proxy }}"{% endif %} {% if https_proxy is defined %}"HTTPS_PROXY={{ https_proxy }}"{% endif %} {% if no_proxy is defined %}"NO_PROXY={{ no_proxy }}"{% endif %} diff --git a/kubespray/roles/container-engine/docker/templates/rh_docker.repo.j2 b/kubespray/roles/container-engine/docker/templates/rh_docker.repo.j2 new file mode 100644 index 0000000..178bbc2 --- /dev/null +++ b/kubespray/roles/container-engine/docker/templates/rh_docker.repo.j2 @@ -0,0 +1,10 @@ +[docker-ce] +name=Docker-CE Repository +baseurl={{ docker_rh_repo_base_url }} +enabled=0 +gpgcheck={{ '1' if docker_rh_repo_gpgkey else '0' }} +keepcache={{ docker_rpm_keepcache | default('1') }} +gpgkey={{ docker_rh_repo_gpgkey }} +{% if http_proxy is defined %} +proxy={{ http_proxy }} +{% endif %} diff --git a/kubespray/roles/container-engine/docker/vars/amazon.yml b/kubespray/roles/container-engine/docker/vars/amazon.yml new file mode 100644 index 0000000..4871f4a --- /dev/null +++ b/kubespray/roles/container-engine/docker/vars/amazon.yml @@ -0,0 +1,15 @@ +--- +# https://docs.aws.amazon.com/en_us/AmazonECS/latest/developerguide/docker-basics.html + +docker_versioned_pkg: + 'latest': docker + '18.09': docker-18.09.9ce-2.amzn2 + '19.03': docker-19.03.13ce-1.amzn2 + '20.10': docker-20.10.7-5.amzn2 + +docker_version: "latest" + +docker_package_info: + pkgs: + - "{{ docker_versioned_pkg[docker_version | string] }}" + enablerepo: amzn2extra-docker diff --git a/kubespray/roles/container-engine/docker/vars/clearlinux.yml b/kubespray/roles/container-engine/docker/vars/clearlinux.yml new file mode 100644 index 0000000..fbb7a22 --- /dev/null +++ b/kubespray/roles/container-engine/docker/vars/clearlinux.yml @@ -0,0 +1,4 @@ +--- +docker_package_info: + pkgs: + - "containers-basic" diff --git a/kubespray/roles/container-engine/docker/vars/debian-bookworm.yml b/kubespray/roles/container-engine/docker/vars/debian-bookworm.yml new file mode 100644 index 0000000..74a66cc --- /dev/null +++ b/kubespray/roles/container-engine/docker/vars/debian-bookworm.yml @@ -0,0 +1,48 @@ +--- +docker_version: 24.0 +docker_cli_version: 24.0 +docker_containerd_version: 1.6.21 + +# containerd package info is only relevant for docker +containerd_versioned_pkg: + 'latest': "{{ containerd_package }}" + '1.6.16': "{{ containerd_package }}=1.6.16-1" + '1.6.18': "{{ containerd_package }}=1.6.18-1" + '1.6.19': "{{ containerd_package }}=1.6.19-1" + '1.6.20': "{{ containerd_package }}=1.6.20-1" + '1.6.21': "{{ containerd_package }}=1.6.21-1" + 'stable': "{{ containerd_package }}=1.6.21-1" + 'edge': "{{ containerd_package }}=1.6.21-1" + +# https://download.docker.com/linux/debian/ +docker_versioned_pkg: + 'latest': docker-ce + '23.0': docker-ce=5:23.0.6-1~debian.{{ ansible_distribution_major_version }}~{{ ansible_distribution_release | lower }} + '24.0': docker-ce=5:24.0.2-1~debian.{{ ansible_distribution_major_version }}~{{ ansible_distribution_release | lower }} + 'stable': docker-ce=5:24.0.2-1~debian.{{ ansible_distribution_major_version }}~{{ ansible_distribution_release | lower }} + 'edge': docker-ce=5:24.0.2-1~debian.{{ ansible_distribution_major_version }}~{{ ansible_distribution_release | lower }} + +docker_cli_versioned_pkg: + 'latest': docker-ce-cli + '23.0': docker-ce=5:23.0.6-1~debian.{{ ansible_distribution_major_version }}~{{ ansible_distribution_release | lower }} + '24.0': docker-ce=5:24.0.2-1~debian.{{ ansible_distribution_major_version }}~{{ ansible_distribution_release | lower }} + 'stable': docker-ce=5:24.0.2-1~debian.{{ ansible_distribution_major_version }}~{{ ansible_distribution_release | lower }} + 'edge': docker-ce=5:24.0.2-1~debian.{{ ansible_distribution_major_version }}~{{ ansible_distribution_release | lower }} + +docker_package_info: + pkgs: + - "{{ containerd_versioned_pkg[docker_containerd_version | string] }}" + - "{{ docker_cli_versioned_pkg[docker_cli_version | string] }}" + - "{{ docker_versioned_pkg[docker_version | string] }}" + +docker_repo_key_info: + url: '{{ docker_debian_repo_gpgkey }}' + repo_keys: + - '{{ docker_debian_repo_repokey }}' + +docker_repo_info: + repos: + - > + deb {{ docker_debian_repo_base_url }} + {{ ansible_distribution_release | lower }} + stable diff --git a/kubespray/roles/container-engine/docker/vars/debian.yml b/kubespray/roles/container-engine/docker/vars/debian.yml new file mode 100644 index 0000000..9f06004 --- /dev/null +++ b/kubespray/roles/container-engine/docker/vars/debian.yml @@ -0,0 +1,61 @@ +--- +# containerd package info is only relevant for docker +containerd_versioned_pkg: + 'latest': "{{ containerd_package }}" + '1.3.7': "{{ containerd_package }}=1.3.7-1" + '1.3.9': "{{ containerd_package }}=1.3.9-1" + '1.4.3': "{{ containerd_package }}=1.4.3-2" + '1.4.4': "{{ containerd_package }}=1.4.4-1" + '1.4.6': "{{ containerd_package }}=1.4.6-1" + '1.4.9': "{{ containerd_package }}=1.4.9-1" + '1.4.12': "{{ containerd_package }}=1.4.12-1" + '1.6.4': "{{ containerd_package }}=1.6.4-1" + '1.6.6': "{{ containerd_package }}=1.6.6-1" + '1.6.7': "{{ containerd_package }}=1.6.7-1" + '1.6.8': "{{ containerd_package }}=1.6.8-1" + '1.6.9': "{{ containerd_package }}=1.6.9-1" + '1.6.10': "{{ containerd_package }}=1.6.10-1" + '1.6.11': "{{ containerd_package }}=1.6.11-1" + '1.6.12': "{{ containerd_package }}=1.6.12-1" + '1.6.13': "{{ containerd_package }}=1.6.13-1" + '1.6.14': "{{ containerd_package }}=1.6.14-1" + '1.6.15': "{{ containerd_package }}=1.6.15-1" + '1.6.16': "{{ containerd_package }}=1.6.16-1" + '1.6.18': "{{ containerd_package }}=1.6.18-1" + 'stable': "{{ containerd_package }}=1.6.18-1" + 'edge': "{{ containerd_package }}=1.6.18-1" + +# https://download.docker.com/linux/debian/ +docker_versioned_pkg: + 'latest': docker-ce + '18.09': docker-ce=5:18.09.9~3-0~debian-{{ ansible_distribution_release | lower }} + '19.03': docker-ce=5:19.03.15~3-0~debian-{{ ansible_distribution_release | lower }} + '20.10': docker-ce=5:20.10.20~3-0~debian-{{ ansible_distribution_release | lower }} + 'stable': docker-ce=5:20.10.20~3-0~debian-{{ ansible_distribution_release | lower }} + 'edge': docker-ce=5:20.10.20~3-0~debian-{{ ansible_distribution_release | lower }} + +docker_cli_versioned_pkg: + 'latest': docker-ce-cli + '18.09': docker-ce-cli=5:18.09.9~3-0~debian-{{ ansible_distribution_release | lower }} + '19.03': docker-ce-cli=5:19.03.15~3-0~debian-{{ ansible_distribution_release | lower }} + '20.10': docker-ce-cli=5:20.10.20~3-0~debian-{{ ansible_distribution_release | lower }} + 'stable': docker-ce-cli=5:20.10.20~3-0~debian-{{ ansible_distribution_release | lower }} + 'edge': docker-ce-cli=5:20.10.20~3-0~debian-{{ ansible_distribution_release | lower }} + +docker_package_info: + pkgs: + - "{{ containerd_versioned_pkg[docker_containerd_version | string] }}" + - "{{ docker_cli_versioned_pkg[docker_cli_version | string] }}" + - "{{ docker_versioned_pkg[docker_version | string] }}" + +docker_repo_key_info: + url: '{{ docker_debian_repo_gpgkey }}' + repo_keys: + - '{{ docker_debian_repo_repokey }}' + +docker_repo_info: + repos: + - > + deb {{ docker_debian_repo_base_url }} + {{ ansible_distribution_release | lower }} + stable diff --git a/kubespray/roles/container-engine/docker/vars/fedora.yml b/kubespray/roles/container-engine/docker/vars/fedora.yml new file mode 100644 index 0000000..f0b7862 --- /dev/null +++ b/kubespray/roles/container-engine/docker/vars/fedora.yml @@ -0,0 +1,49 @@ +--- +# containerd versions are only relevant for docker +containerd_versioned_pkg: + 'latest': "{{ containerd_package }}" + '1.3.7': "{{ containerd_package }}-1.3.7-3.1.fc{{ ansible_distribution_major_version }}" + '1.3.9': "{{ containerd_package }}-1.3.9-3.1.fc{{ ansible_distribution_major_version }}" + '1.4.3': "{{ containerd_package }}-1.4.3-3.2.fc{{ ansible_distribution_major_version }}" + '1.4.4': "{{ containerd_package }}-1.4.4-3.1.fc{{ ansible_distribution_major_version }}" + '1.4.6': "{{ containerd_package }}-1.4.6-3.1.fc{{ ansible_distribution_major_version }}" + '1.4.9': "{{ containerd_package }}-1.4.9-3.1.fc{{ ansible_distribution_major_version }}" + '1.4.12': "{{ containerd_package }}-1.4.12-3.1.fc{{ ansible_distribution_major_version }}" + '1.6.4': "{{ containerd_package }}-1.6.4-3.1.fc{{ ansible_distribution_major_version }}" + '1.6.6': "{{ containerd_package }}-1.6.6-3.1.fc{{ ansible_distribution_major_version }}" + '1.6.7': "{{ containerd_package }}-1.6.7-3.1.fc{{ ansible_distribution_major_version }}" + '1.6.8': "{{ containerd_package }}-1.6.8-3.1.fc{{ ansible_distribution_major_version }}" + '1.6.9': "{{ containerd_package }}-1.6.9-3.1.fc{{ ansible_distribution_major_version }}" + '1.6.10': "{{ containerd_package }}-1.6.10-3.1.fc{{ ansible_distribution_major_version }}" + '1.6.11': "{{ containerd_package }}-1.6.11-3.1.fc{{ ansible_distribution_major_version }}" + '1.6.12': "{{ containerd_package }}-1.6.12-3.1.fc{{ ansible_distribution_major_version }}" + '1.6.13': "{{ containerd_package }}-1.6.13-3.1.fc{{ ansible_distribution_major_version }}" + '1.6.14': "{{ containerd_package }}-1.6.14-3.1.fc{{ ansible_distribution_major_version }}" + '1.6.15': "{{ containerd_package }}-1.6.15-3.1.fc{{ ansible_distribution_major_version }}" + '1.6.16': "{{ containerd_package }}-1.6.16-3.1.fc{{ ansible_distribution_major_version }}" + '1.6.18': "{{ containerd_package }}-1.6.18-3.1.fc{{ ansible_distribution_major_version }}" + 'stable': "{{ containerd_package }}-1.6.18-3.1.fc{{ ansible_distribution_major_version }}" + 'edge': "{{ containerd_package }}-1.6.18-3.1.fc{{ ansible_distribution_major_version }}" + +# https://docs.docker.com/install/linux/docker-ce/fedora/ +# https://download.docker.com/linux/fedora//x86_64/stable/Packages/ +docker_versioned_pkg: + 'latest': docker-ce + '19.03': docker-ce-19.03.15-3.fc{{ ansible_distribution_major_version }} + '20.10': docker-ce-20.10.20-3.fc{{ ansible_distribution_major_version }} + 'stable': docker-ce-20.10.20-3.fc{{ ansible_distribution_major_version }} + 'edge': docker-ce-20.10.20-3.fc{{ ansible_distribution_major_version }} + +docker_cli_versioned_pkg: + 'latest': docker-ce-cli + '19.03': docker-ce-cli-19.03.15-3.fc{{ ansible_distribution_major_version }} + '20.10': docker-ce-cli-20.10.20-3.fc{{ ansible_distribution_major_version }} + 'stable': docker-ce-cli-20.10.20-3.fc{{ ansible_distribution_major_version }} + 'edge': docker-ce-cli-20.10.20-3.fc{{ ansible_distribution_major_version }} + +docker_package_info: + enablerepo: "docker-ce" + pkgs: + - "{{ containerd_versioned_pkg[docker_containerd_version | string] }}" + - "{{ docker_cli_versioned_pkg[docker_cli_version | string] }}" + - "{{ docker_versioned_pkg[docker_version | string] }}" diff --git a/kubespray/roles/container-engine/docker/vars/kylin.yml b/kubespray/roles/container-engine/docker/vars/kylin.yml new file mode 100644 index 0000000..b933f15 --- /dev/null +++ b/kubespray/roles/container-engine/docker/vars/kylin.yml @@ -0,0 +1,53 @@ +--- +# containerd versions are only relevant for docker +containerd_versioned_pkg: + 'latest': "{{ containerd_package }}" + '1.3.7': "{{ containerd_package }}-1.3.7-3.1.el{{ ansible_distribution_major_version }}" + '1.3.9': "{{ containerd_package }}-1.3.9-3.1.el{{ ansible_distribution_major_version }}" + '1.4.3': "{{ containerd_package }}-1.4.3-3.2.el{{ ansible_distribution_major_version }}" + '1.4.4': "{{ containerd_package }}-1.4.4-3.1.el{{ ansible_distribution_major_version }}" + '1.4.6': "{{ containerd_package }}-1.4.6-3.1.el{{ ansible_distribution_major_version }}" + '1.4.9': "{{ containerd_package }}-1.4.9-3.1.el{{ ansible_distribution_major_version }}" + '1.4.12': "{{ containerd_package }}-1.4.12-3.1.el{{ ansible_distribution_major_version }}" + '1.6.4': "{{ containerd_package }}-1.6.4-3.1.el{{ ansible_distribution_major_version }}" + '1.6.6': "{{ containerd_package }}-1.6.6-3.1.el{{ ansible_distribution_major_version }}" + '1.6.7': "{{ containerd_package }}-1.6.7-3.1.el{{ ansible_distribution_major_version }}" + '1.6.8': "{{ containerd_package }}-1.6.8-3.1.el{{ ansible_distribution_major_version }}" + '1.6.9': "{{ containerd_package }}-1.6.9-3.1.el{{ ansible_distribution_major_version }}" + '1.6.10': "{{ containerd_package }}-1.6.10-3.1.el{{ ansible_distribution_major_version }}" + '1.6.11': "{{ containerd_package }}-1.6.11-3.1.el{{ ansible_distribution_major_version }}" + '1.6.12': "{{ containerd_package }}-1.6.12-3.1.el{{ ansible_distribution_major_version }}" + '1.6.13': "{{ containerd_package }}-1.6.13-3.1.el{{ ansible_distribution_major_version }}" + '1.6.14': "{{ containerd_package }}-1.6.14-3.1.el{{ ansible_distribution_major_version }}" + '1.6.15': "{{ containerd_package }}-1.6.15-3.1.el{{ ansible_distribution_major_version }}" + '1.6.16': "{{ containerd_package }}-1.6.16-3.1.el{{ ansible_distribution_major_version }}" + '1.6.18': "{{ containerd_package }}-1.6.18-3.1.el{{ ansible_distribution_major_version }}" + 'stable': "{{ containerd_package }}-1.6.18-3.1.el{{ ansible_distribution_major_version }}" + 'edge': "{{ containerd_package }}-1.6.18-3.1.el{{ ansible_distribution_major_version }}" + +docker_version: 19.03 +docker_cli_version: 19.03 + +# https://docs.docker.com/engine/installation/linux/centos/#install-from-a-package +# https://download.docker.com/linux/centos/>/x86_64/stable/Packages/ +# or do 'yum --showduplicates list docker-engine' +docker_versioned_pkg: + 'latest': docker-ce + '18.09': docker-ce-3:18.09.9-3.el7 + '19.03': docker-ce-3:19.03.15-3.el{{ ansible_distribution_major_version }} + 'stable': docker-ce-3:19.03.15-3.el{{ ansible_distribution_major_version }} + 'edge': docker-ce-3:19.03.15-3.el{{ ansible_distribution_major_version }} + +docker_cli_versioned_pkg: + 'latest': docker-ce-cli + '18.09': docker-ce-cli-1:18.09.9-3.el7 + '19.03': docker-ce-cli-1:19.03.15-3.el{{ ansible_distribution_major_version }} + 'stable': docker-ce-cli-1:19.03.15-3.el{{ ansible_distribution_major_version }} + 'edge': docker-ce-cli-1:19.03.15-3.el{{ ansible_distribution_major_version }} + +docker_package_info: + enablerepo: "docker-ce" + pkgs: + - "{{ containerd_versioned_pkg[docker_containerd_version | string] }}" + - "{{ docker_cli_versioned_pkg[docker_cli_version | string] }}" + - "{{ docker_versioned_pkg[docker_version | string] }}" diff --git a/kubespray/roles/container-engine/docker/vars/redhat-7.yml b/kubespray/roles/container-engine/docker/vars/redhat-7.yml new file mode 100644 index 0000000..f50d99d --- /dev/null +++ b/kubespray/roles/container-engine/docker/vars/redhat-7.yml @@ -0,0 +1,52 @@ +--- +# containerd versions are only relevant for docker +containerd_versioned_pkg: + 'latest': "{{ containerd_package }}" + '1.3.7': "{{ containerd_package }}-1.3.7-3.1.el7" + '1.3.9': "{{ containerd_package }}-1.3.9-3.1.el7" + '1.4.3': "{{ containerd_package }}-1.4.3-3.2.el7" + '1.4.4': "{{ containerd_package }}-1.4.4-3.1.el7" + '1.4.6': "{{ containerd_package }}-1.4.6-3.1.el7" + '1.4.9': "{{ containerd_package }}-1.4.9-3.1.el7" + '1.4.12': "{{ containerd_package }}-1.4.12-3.1.el7" + '1.6.4': "{{ containerd_package }}-1.6.4-3.1.el7" + '1.6.6': "{{ containerd_package }}-1.6.6-3.1.el7" + '1.6.7': "{{ containerd_package }}-1.6.7-3.1.el7" + '1.6.8': "{{ containerd_package }}-1.6.8-3.1.el7" + '1.6.9': "{{ containerd_package }}-1.6.9-3.1.el7" + '1.6.10': "{{ containerd_package }}-1.6.10-3.1.el7" + '1.6.11': "{{ containerd_package }}-1.6.11-3.1.el7" + '1.6.12': "{{ containerd_package }}-1.6.12-3.1.el7" + '1.6.13': "{{ containerd_package }}-1.6.13-3.1.el7" + '1.6.14': "{{ containerd_package }}-1.6.14-3.1.el7" + '1.6.15': "{{ containerd_package }}-1.6.15-3.1.el7" + '1.6.16': "{{ containerd_package }}-1.6.16-3.1.el7" + '1.6.18': "{{ containerd_package }}-1.6.18-3.1.el7" + 'stable': "{{ containerd_package }}-1.6.18-3.1.el7" + 'edge': "{{ containerd_package }}-1.6.18-3.1.el7" + +# https://docs.docker.com/engine/installation/linux/centos/#install-from-a-package +# https://download.docker.com/linux/centos/>/x86_64/stable/Packages/ +# or do 'yum --showduplicates list docker-engine' +docker_versioned_pkg: + 'latest': docker-ce + '18.09': docker-ce-18.09.9-3.el7 + '19.03': docker-ce-19.03.15-3.el7 + '20.10': docker-ce-20.10.20-3.el7 + 'stable': docker-ce-20.10.20-3.el7 + 'edge': docker-ce-20.10.20-3.el7 + +docker_cli_versioned_pkg: + 'latest': docker-ce-cli + '18.09': docker-ce-cli-18.09.9-3.el7 + '19.03': docker-ce-cli-19.03.15-3.el7 + '20.10': docker-ce-cli-20.10.20-3.el7 + 'stable': docker-ce-cli-20.10.20-3.el7 + 'edge': docker-ce-cli-20.10.20-3.el7 + +docker_package_info: + enablerepo: "docker-ce" + pkgs: + - "{{ containerd_versioned_pkg[docker_containerd_version | string] }}" + - "{{ docker_cli_versioned_pkg[docker_cli_version | string] }}" + - "{{ docker_versioned_pkg[docker_version | string] }}" diff --git a/kubespray/roles/container-engine/docker/vars/redhat.yml b/kubespray/roles/container-engine/docker/vars/redhat.yml new file mode 100644 index 0000000..1de2cbe --- /dev/null +++ b/kubespray/roles/container-engine/docker/vars/redhat.yml @@ -0,0 +1,52 @@ +--- +# containerd versions are only relevant for docker +containerd_versioned_pkg: + 'latest': "{{ containerd_package }}" + '1.3.7': "{{ containerd_package }}-1.3.7-3.1.el{{ ansible_distribution_major_version }}" + '1.3.9': "{{ containerd_package }}-1.3.9-3.1.el{{ ansible_distribution_major_version }}" + '1.4.3': "{{ containerd_package }}-1.4.3-3.2.el{{ ansible_distribution_major_version }}" + '1.4.4': "{{ containerd_package }}-1.4.4-3.1.el{{ ansible_distribution_major_version }}" + '1.4.6': "{{ containerd_package }}-1.4.6-3.1.el{{ ansible_distribution_major_version }}" + '1.4.9': "{{ containerd_package }}-1.4.9-3.1.el{{ ansible_distribution_major_version }}" + '1.4.12': "{{ containerd_package }}-1.4.12-3.1.el{{ ansible_distribution_major_version }}" + '1.6.4': "{{ containerd_package }}-1.6.4-3.1.el{{ ansible_distribution_major_version }}" + '1.6.6': "{{ containerd_package }}-1.6.6-3.1.el{{ ansible_distribution_major_version }}" + '1.6.7': "{{ containerd_package }}-1.6.7-3.1.el{{ ansible_distribution_major_version }}" + '1.6.8': "{{ containerd_package }}-1.6.8-3.1.el{{ ansible_distribution_major_version }}" + '1.6.9': "{{ containerd_package }}-1.6.9-3.1.el{{ ansible_distribution_major_version }}" + '1.6.10': "{{ containerd_package }}-1.6.10-3.1.el{{ ansible_distribution_major_version }}" + '1.6.11': "{{ containerd_package }}-1.6.11-3.1.el{{ ansible_distribution_major_version }}" + '1.6.12': "{{ containerd_package }}-1.6.12-3.1.el{{ ansible_distribution_major_version }}" + '1.6.13': "{{ containerd_package }}-1.6.13-3.1.el{{ ansible_distribution_major_version }}" + '1.6.14': "{{ containerd_package }}-1.6.14-3.1.el{{ ansible_distribution_major_version }}" + '1.6.15': "{{ containerd_package }}-1.6.15-3.1.el{{ ansible_distribution_major_version }}" + '1.6.16': "{{ containerd_package }}-1.6.16-3.1.el{{ ansible_distribution_major_version }}" + '1.6.18': "{{ containerd_package }}-1.6.18-3.1.el{{ ansible_distribution_major_version }}" + 'stable': "{{ containerd_package }}-1.6.18-3.1.el{{ ansible_distribution_major_version }}" + 'edge': "{{ containerd_package }}-1.6.18-3.1.el{{ ansible_distribution_major_version }}" + +# https://docs.docker.com/engine/installation/linux/centos/#install-from-a-package +# https://download.docker.com/linux/centos/>/x86_64/stable/Packages/ +# or do 'yum --showduplicates list docker-engine' +docker_versioned_pkg: + 'latest': docker-ce + '18.09': docker-ce-3:18.09.9-3.el7 + '19.03': docker-ce-3:19.03.15-3.el{{ ansible_distribution_major_version }} + '20.10': docker-ce-3:20.10.20-3.el{{ ansible_distribution_major_version }} + 'stable': docker-ce-3:20.10.20-3.el{{ ansible_distribution_major_version }} + 'edge': docker-ce-3:20.10.20-3.el{{ ansible_distribution_major_version }} + +docker_cli_versioned_pkg: + 'latest': docker-ce-cli + '18.09': docker-ce-cli-1:18.09.9-3.el7 + '19.03': docker-ce-cli-1:19.03.15-3.el{{ ansible_distribution_major_version }} + '20.10': docker-ce-cli-1:20.10.20-3.el{{ ansible_distribution_major_version }} + 'stable': docker-ce-cli-1:20.10.20-3.el{{ ansible_distribution_major_version }} + 'edge': docker-ce-cli-1:20.10.20-3.el{{ ansible_distribution_major_version }} + +docker_package_info: + enablerepo: "docker-ce" + pkgs: + - "{{ containerd_versioned_pkg[docker_containerd_version | string] }}" + - "{{ docker_cli_versioned_pkg[docker_cli_version | string] }}" + - "{{ docker_versioned_pkg[docker_version | string] }}" diff --git a/kubespray/roles/container-engine/docker/vars/suse.yml b/kubespray/roles/container-engine/docker/vars/suse.yml new file mode 100644 index 0000000..2d9fbf0 --- /dev/null +++ b/kubespray/roles/container-engine/docker/vars/suse.yml @@ -0,0 +1,6 @@ +--- +docker_package_info: + state: latest + pkgs: + - docker + - containerd diff --git a/kubespray/roles/container-engine/docker/vars/ubuntu.yml b/kubespray/roles/container-engine/docker/vars/ubuntu.yml new file mode 100644 index 0000000..313849e --- /dev/null +++ b/kubespray/roles/container-engine/docker/vars/ubuntu.yml @@ -0,0 +1,61 @@ +--- +# containerd versions are only relevant for docker +containerd_versioned_pkg: + 'latest': "{{ containerd_package }}" + '1.3.7': "{{ containerd_package }}=1.3.7-1" + '1.3.9': "{{ containerd_package }}=1.3.9-1" + '1.4.3': "{{ containerd_package }}=1.4.3-2" + '1.4.4': "{{ containerd_package }}=1.4.4-1" + '1.4.6': "{{ containerd_package }}=1.4.6-1" + '1.4.9': "{{ containerd_package }}=1.4.9-1" + '1.4.12': "{{ containerd_package }}=1.4.12-1" + '1.6.4': "{{ containerd_package }}=1.6.4-1" + '1.6.6': "{{ containerd_package }}=1.6.6-1" + '1.6.7': "{{ containerd_package }}=1.6.7-1" + '1.6.8': "{{ containerd_package }}=1.6.8-1" + '1.6.9': "{{ containerd_package }}=1.6.9-1" + '1.6.10': "{{ containerd_package }}=1.6.10-1" + '1.6.11': "{{ containerd_package }}=1.6.11-1" + '1.6.12': "{{ containerd_package }}=1.6.12-1" + '1.6.13': "{{ containerd_package }}=1.6.13-1" + '1.6.14': "{{ containerd_package }}=1.6.14-1" + '1.6.15': "{{ containerd_package }}=1.6.15-1" + '1.6.16': "{{ containerd_package }}=1.6.16-1" + '1.6.18': "{{ containerd_package }}=1.6.18-1" + 'stable': "{{ containerd_package }}=1.6.18-1" + 'edge': "{{ containerd_package }}=1.6.18-1" + +# https://download.docker.com/linux/ubuntu/ +docker_versioned_pkg: + 'latest': docker-ce + '18.09': docker-ce=5:18.09.9~3-0~ubuntu-{{ ansible_distribution_release | lower }} + '19.03': docker-ce=5:19.03.15~3-0~ubuntu-{{ ansible_distribution_release | lower }} + '20.10': docker-ce=5:20.10.20~3-0~ubuntu-{{ ansible_distribution_release | lower }} + 'stable': docker-ce=5:20.10.20~3-0~ubuntu-{{ ansible_distribution_release | lower }} + 'edge': docker-ce=5:20.10.20~3-0~ubuntu-{{ ansible_distribution_release | lower }} + +docker_cli_versioned_pkg: + 'latest': docker-ce-cli + '18.09': docker-ce-cli=5:18.09.9~3-0~ubuntu-{{ ansible_distribution_release | lower }} + '19.03': docker-ce-cli=5:19.03.15~3-0~ubuntu-{{ ansible_distribution_release | lower }} + '20.10': docker-ce-cli=5:20.10.20~3-0~ubuntu-{{ ansible_distribution_release | lower }} + 'stable': docker-ce-cli=5:20.10.20~3-0~ubuntu-{{ ansible_distribution_release | lower }} + 'edge': docker-ce-cli=5:20.10.20~3-0~ubuntu-{{ ansible_distribution_release | lower }} + +docker_package_info: + pkgs: + - "{{ containerd_versioned_pkg[docker_containerd_version | string] }}" + - "{{ docker_cli_versioned_pkg[docker_cli_version | string] }}" + - "{{ docker_versioned_pkg[docker_version | string] }}" + +docker_repo_key_info: + url: '{{ docker_ubuntu_repo_gpgkey }}' + repo_keys: + - '{{ docker_ubuntu_repo_repokey }}' + +docker_repo_info: + repos: + - > + deb [arch={{ host_architecture }}] {{ docker_ubuntu_repo_base_url }} + {{ ansible_distribution_release | lower }} + stable diff --git a/kubespray/roles/container-engine/docker/vars/uniontech.yml b/kubespray/roles/container-engine/docker/vars/uniontech.yml new file mode 100644 index 0000000..d41cb3b --- /dev/null +++ b/kubespray/roles/container-engine/docker/vars/uniontech.yml @@ -0,0 +1,54 @@ +--- +# containerd versions are only relevant for docker +containerd_versioned_pkg: + 'latest': "{{ containerd_package }}" + '1.3.7': "{{ containerd_package }}-1.3.7-3.1.el{{ ansible_distribution_major_version }}" + '1.3.9': "{{ containerd_package }}-1.3.9-3.1.el{{ ansible_distribution_major_version }}" + '1.4.3': "{{ containerd_package }}-1.4.3-3.2.el{{ ansible_distribution_major_version }}" + '1.4.4': "{{ containerd_package }}-1.4.4-3.1.el{{ ansible_distribution_major_version }}" + '1.4.6': "{{ containerd_package }}-1.4.6-3.1.el{{ ansible_distribution_major_version }}" + '1.4.9': "{{ containerd_package }}-1.4.9-3.1.el{{ ansible_distribution_major_version }}" + '1.4.12': "{{ containerd_package }}-1.4.12-3.1.el{{ ansible_distribution_major_version }}" + '1.6.4': "{{ containerd_package }}-1.6.4-3.1.el{{ ansible_distribution_major_version }}" + '1.6.8': "{{ containerd_package }}-1.6.8-3.1.el{{ ansible_distribution_major_version }}" + '1.6.9': "{{ containerd_package }}-1.6.9-3.1.el{{ ansible_distribution_major_version }}" + '1.6.10': "{{ containerd_package }}-1.6.10-3.1.el{{ ansible_distribution_major_version }}" + '1.6.11': "{{ containerd_package }}-1.6.11-3.1.el{{ ansible_distribution_major_version }}" + '1.6.12': "{{ containerd_package }}-1.6.12-3.1.el{{ ansible_distribution_major_version }}" + '1.6.13': "{{ containerd_package }}-1.6.13-3.1.el{{ ansible_distribution_major_version }}" + '1.6.14': "{{ containerd_package }}-1.6.14-3.1.el{{ ansible_distribution_major_version }}" + '1.6.15': "{{ containerd_package }}-1.6.15-3.1.el{{ ansible_distribution_major_version }}" + '1.6.16': "{{ containerd_package }}-1.6.16-3.1.el{{ ansible_distribution_major_version }}" + '1.6.18': "{{ containerd_package }}-1.6.18-3.1.el{{ ansible_distribution_major_version }}" + 'stable': "{{ containerd_package }}-1.6.18-3.1.el{{ ansible_distribution_major_version }}" + 'edge': "{{ containerd_package }}-1.6.18-3.1.el{{ ansible_distribution_major_version }}" + +docker_version: 19.03 +docker_cli_version: 19.03 + +# https://docs.docker.com/engine/installation/linux/centos/#install-from-a-package +# https://download.docker.com/linux/centos/>/x86_64/stable/Packages/ +# or do 'yum --showduplicates list docker-engine' +docker_versioned_pkg: + 'latest': docker-ce + '18.09': docker-ce-3:18.09.9-3.el7 + '19.03': docker-ce-3:19.03.15-3.el{{ ansible_distribution_major_version }} + '20.10': docker-ce-3:20.10.17-3.el{{ ansible_distribution_major_version }} + 'stable': docker-ce-3:20.10.17-3.el{{ ansible_distribution_major_version }} + 'edge': docker-ce-3:20.10.17-3.el{{ ansible_distribution_major_version }} + +docker_cli_versioned_pkg: + 'latest': docker-ce-cli + '18.09': docker-ce-cli-1:18.09.9-3.el7 + '19.03': docker-ce-cli-1:19.03.15-3.el{{ ansible_distribution_major_version }} + '20.10': docker-ce-cli-1:20.10.17-3.el{{ ansible_distribution_major_version }} + 'stable': docker-ce-cli-1:20.10.17-3.el{{ ansible_distribution_major_version }} + 'edge': docker-ce-cli-1:20.10.17-3.el{{ ansible_distribution_major_version }} + +docker_package_info: + enablerepo: "docker-ce" + disablerepo: "UniontechOS-20-AppStream" + pkgs: + - "{{ containerd_versioned_pkg[docker_containerd_version | string] }}" + - "{{ docker_cli_versioned_pkg[docker_cli_version | string] }}" + - "{{ docker_versioned_pkg[docker_version | string] }}" diff --git a/kubespray/roles/container-engine/gvisor/molecule/default/converge.yml b/kubespray/roles/container-engine/gvisor/molecule/default/converge.yml new file mode 100644 index 0000000..b14d078 --- /dev/null +++ b/kubespray/roles/container-engine/gvisor/molecule/default/converge.yml @@ -0,0 +1,11 @@ +--- +- name: Converge + hosts: all + become: true + vars: + gvisor_enabled: true + container_manager: containerd + roles: + - role: kubespray-defaults + - role: container-engine/containerd + - role: container-engine/gvisor diff --git a/kubespray/roles/container-engine/gvisor/molecule/default/files/10-mynet.conf b/kubespray/roles/container-engine/gvisor/molecule/default/files/10-mynet.conf new file mode 100644 index 0000000..f10935b --- /dev/null +++ b/kubespray/roles/container-engine/gvisor/molecule/default/files/10-mynet.conf @@ -0,0 +1,17 @@ +{ + "cniVersion": "0.2.0", + "name": "mynet", + "type": "bridge", + "bridge": "cni0", + "isGateway": true, + "ipMasq": true, + "ipam": { + "type": "host-local", + "subnet": "172.19.0.0/24", + "routes": [ + { + "dst": "0.0.0.0/0" + } + ] + } +} diff --git a/kubespray/roles/container-engine/gvisor/molecule/default/files/container.json b/kubespray/roles/container-engine/gvisor/molecule/default/files/container.json new file mode 100644 index 0000000..acec0ce --- /dev/null +++ b/kubespray/roles/container-engine/gvisor/molecule/default/files/container.json @@ -0,0 +1,10 @@ +{ + "metadata": { + "name": "gvisor1" + }, + "image": { + "image": "quay.io/kubespray/hello-world:latest" + }, + "log_path": "gvisor1.0.log", + "linux": {} +} diff --git a/kubespray/roles/container-engine/gvisor/molecule/default/files/sandbox.json b/kubespray/roles/container-engine/gvisor/molecule/default/files/sandbox.json new file mode 100644 index 0000000..a8da54d --- /dev/null +++ b/kubespray/roles/container-engine/gvisor/molecule/default/files/sandbox.json @@ -0,0 +1,10 @@ +{ + "metadata": { + "name": "gvisor1", + "namespace": "default", + "attempt": 1, + "uid": "hdishd83djaidwnduwk28bcsb" + }, + "linux": {}, + "log_directory": "/tmp" +} diff --git a/kubespray/roles/container-engine/gvisor/molecule/default/molecule.yml b/kubespray/roles/container-engine/gvisor/molecule/default/molecule.yml new file mode 100644 index 0000000..9ba1927 --- /dev/null +++ b/kubespray/roles/container-engine/gvisor/molecule/default/molecule.yml @@ -0,0 +1,39 @@ +--- +role_name_check: 1 +driver: + name: vagrant + provider: + name: libvirt +platforms: + - name: ubuntu20 + box: generic/ubuntu2004 + cpus: 1 + memory: 1024 + nested: true + groups: + - kube_control_plane + provider_options: + driver: kvm + - name: almalinux8 + box: almalinux/8 + cpus: 1 + memory: 1024 + nested: true + groups: + - kube_control_plane + provider_options: + driver: kvm +provisioner: + name: ansible + env: + ANSIBLE_ROLES_PATH: ../../../../ + config_options: + defaults: + callbacks_enabled: profile_tasks + timeout: 120 + inventory: + group_vars: + all: + become: true +verifier: + name: testinfra diff --git a/kubespray/roles/container-engine/gvisor/molecule/default/prepare.yml b/kubespray/roles/container-engine/gvisor/molecule/default/prepare.yml new file mode 100644 index 0000000..3ec3602 --- /dev/null +++ b/kubespray/roles/container-engine/gvisor/molecule/default/prepare.yml @@ -0,0 +1,49 @@ +--- +- name: Prepare generic + hosts: all + become: true + roles: + - role: kubespray-defaults + - role: bootstrap-os + - role: adduser + user: "{{ addusers.kube }}" + tasks: + - name: Download CNI + include_tasks: "../../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.cni) }}" + +- name: Prepare container runtime + hosts: all + become: true + vars: + container_manager: containerd + kube_network_plugin: cni + roles: + - role: kubespray-defaults + - role: network_plugin/cni + - role: container-engine/crictl + tasks: + - name: Copy test container files + copy: + src: "{{ item }}" + dest: "/tmp/{{ item }}" + owner: root + mode: 0644 + with_items: + - container.json + - sandbox.json + - name: Create /etc/cni/net.d directory + file: + path: /etc/cni/net.d + state: directory + owner: root + mode: 0755 + - name: Setup CNI + copy: + src: "{{ item }}" + dest: "/etc/cni/net.d/{{ item }}" + owner: root + mode: 0644 + with_items: + - 10-mynet.conf diff --git a/kubespray/roles/container-engine/gvisor/molecule/default/tests/test_default.py b/kubespray/roles/container-engine/gvisor/molecule/default/tests/test_default.py new file mode 100644 index 0000000..1cb7fb0 --- /dev/null +++ b/kubespray/roles/container-engine/gvisor/molecule/default/tests/test_default.py @@ -0,0 +1,29 @@ +import os + +import testinfra.utils.ansible_runner + +testinfra_hosts = testinfra.utils.ansible_runner.AnsibleRunner( + os.environ['MOLECULE_INVENTORY_FILE']).get_hosts('all') + + +def test_run(host): + gvisorruntime = "/usr/local/bin/runsc" + with host.sudo(): + cmd = host.command(gvisorruntime + " --version") + assert cmd.rc == 0 + assert "runsc version" in cmd.stdout + + +def test_run_pod(host): + runtime = "runsc" + + run_command = "/usr/local/bin/crictl run --with-pull --runtime {} /tmp/container.json /tmp/sandbox.json".format(runtime) + with host.sudo(): + cmd = host.command(run_command) + assert cmd.rc == 0 + + with host.sudo(): + log_f = host.file("/tmp/gvisor1.0.log") + + assert log_f.exists + assert b"Hello from Docker" in log_f.content diff --git a/kubespray/roles/container-engine/gvisor/tasks/main.yml b/kubespray/roles/container-engine/gvisor/tasks/main.yml new file mode 100644 index 0000000..1a8277b --- /dev/null +++ b/kubespray/roles/container-engine/gvisor/tasks/main.yml @@ -0,0 +1,20 @@ +--- +- name: GVisor | Download runsc binary + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.gvisor_runsc) }}" + +- name: GVisor | Download containerd-shim-runsc-v1 binary + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.gvisor_containerd_shim) }}" + +- name: GVisor | Copy binaries + copy: + src: "{{ item.src }}" + dest: "{{ bin_dir }}/{{ item.dest }}" + mode: 0755 + remote_src: yes + with_items: + - { src: "{{ downloads.gvisor_runsc.dest }}", dest: "runsc" } + - { src: "{{ downloads.gvisor_containerd_shim.dest }}", dest: "containerd-shim-runsc-v1" } diff --git a/kubespray/roles/container-engine/kata-containers/OWNERS b/kubespray/roles/container-engine/kata-containers/OWNERS new file mode 100644 index 0000000..fa95926 --- /dev/null +++ b/kubespray/roles/container-engine/kata-containers/OWNERS @@ -0,0 +1,6 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +approvers: + - pasqualet +reviewers: + - pasqualet diff --git a/kubespray/roles/container-engine/kata-containers/defaults/main.yml b/kubespray/roles/container-engine/kata-containers/defaults/main.yml new file mode 100644 index 0000000..40bbc33 --- /dev/null +++ b/kubespray/roles/container-engine/kata-containers/defaults/main.yml @@ -0,0 +1,10 @@ +--- +kata_containers_dir: /opt/kata +kata_containers_config_dir: /etc/kata-containers +kata_containers_containerd_bin_dir: /usr/local/bin + +kata_containers_qemu_default_memory: "{{ ansible_memtotal_mb }}" +kata_containers_qemu_debug: 'false' +kata_containers_qemu_sandbox_cgroup_only: 'true' +kata_containers_qemu_enable_mem_prealloc: 'false' +kata_containers_virtio_fs_cache: 'always' diff --git a/kubespray/roles/container-engine/kata-containers/molecule/default/converge.yml b/kubespray/roles/container-engine/kata-containers/molecule/default/converge.yml new file mode 100644 index 0000000..a6fdf81 --- /dev/null +++ b/kubespray/roles/container-engine/kata-containers/molecule/default/converge.yml @@ -0,0 +1,11 @@ +--- +- name: Converge + hosts: all + become: true + vars: + kata_containers_enabled: true + container_manager: containerd + roles: + - role: kubespray-defaults + - role: container-engine/containerd + - role: container-engine/kata-containers diff --git a/kubespray/roles/container-engine/kata-containers/molecule/default/files/10-mynet.conf b/kubespray/roles/container-engine/kata-containers/molecule/default/files/10-mynet.conf new file mode 100644 index 0000000..f10935b --- /dev/null +++ b/kubespray/roles/container-engine/kata-containers/molecule/default/files/10-mynet.conf @@ -0,0 +1,17 @@ +{ + "cniVersion": "0.2.0", + "name": "mynet", + "type": "bridge", + "bridge": "cni0", + "isGateway": true, + "ipMasq": true, + "ipam": { + "type": "host-local", + "subnet": "172.19.0.0/24", + "routes": [ + { + "dst": "0.0.0.0/0" + } + ] + } +} diff --git a/kubespray/roles/container-engine/kata-containers/molecule/default/files/container.json b/kubespray/roles/container-engine/kata-containers/molecule/default/files/container.json new file mode 100644 index 0000000..e2e9a56 --- /dev/null +++ b/kubespray/roles/container-engine/kata-containers/molecule/default/files/container.json @@ -0,0 +1,10 @@ +{ + "metadata": { + "name": "kata1" + }, + "image": { + "image": "quay.io/kubespray/hello-world:latest" + }, + "log_path": "kata1.0.log", + "linux": {} +} diff --git a/kubespray/roles/container-engine/kata-containers/molecule/default/files/sandbox.json b/kubespray/roles/container-engine/kata-containers/molecule/default/files/sandbox.json new file mode 100644 index 0000000..326a578 --- /dev/null +++ b/kubespray/roles/container-engine/kata-containers/molecule/default/files/sandbox.json @@ -0,0 +1,10 @@ +{ + "metadata": { + "name": "kata1", + "namespace": "default", + "attempt": 1, + "uid": "hdishd83djaidwnduwk28bcsb" + }, + "linux": {}, + "log_directory": "/tmp" +} diff --git a/kubespray/roles/container-engine/kata-containers/molecule/default/molecule.yml b/kubespray/roles/container-engine/kata-containers/molecule/default/molecule.yml new file mode 100644 index 0000000..8eaa5d7 --- /dev/null +++ b/kubespray/roles/container-engine/kata-containers/molecule/default/molecule.yml @@ -0,0 +1,39 @@ +--- +role_name_check: 1 +driver: + name: vagrant + provider: + name: libvirt +platforms: + - name: ubuntu20 + box: generic/ubuntu2004 + cpus: 1 + memory: 1024 + nested: true + groups: + - kube_control_plane + provider_options: + driver: kvm + - name: ubuntu22 + box: generic/ubuntu2204 + cpus: 1 + memory: 1024 + nested: true + groups: + - kube_control_plane + provider_options: + driver: kvm +provisioner: + name: ansible + env: + ANSIBLE_ROLES_PATH: ../../../../ + config_options: + defaults: + callbacks_enabled: profile_tasks + timeout: 120 + inventory: + group_vars: + all: + become: true +verifier: + name: testinfra diff --git a/kubespray/roles/container-engine/kata-containers/molecule/default/prepare.yml b/kubespray/roles/container-engine/kata-containers/molecule/default/prepare.yml new file mode 100644 index 0000000..9d7019a --- /dev/null +++ b/kubespray/roles/container-engine/kata-containers/molecule/default/prepare.yml @@ -0,0 +1,49 @@ +--- +- name: Prepare + hosts: all + become: true + roles: + - role: kubespray-defaults + - role: bootstrap-os + - role: adduser + user: "{{ addusers.kube }}" + tasks: + - name: Download CNI + include_tasks: "../../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.cni) }}" + +- name: Prepare container runtime + hosts: all + become: true + vars: + container_manager: containerd + kube_network_plugin: cni + roles: + - role: kubespray-defaults + - role: network_plugin/cni + - role: container-engine/crictl + tasks: + - name: Copy test container files + copy: + src: "{{ item }}" + dest: "/tmp/{{ item }}" + owner: root + mode: 0644 + with_items: + - container.json + - sandbox.json + - name: Create /etc/cni/net.d directory + file: + path: /etc/cni/net.d + state: directory + owner: "{{ kube_owner }}" + mode: 0755 + - name: Setup CNI + copy: + src: "{{ item }}" + dest: "/etc/cni/net.d/{{ item }}" + owner: root + mode: 0644 + with_items: + - 10-mynet.conf diff --git a/kubespray/roles/container-engine/kata-containers/molecule/default/tests/test_default.py b/kubespray/roles/container-engine/kata-containers/molecule/default/tests/test_default.py new file mode 100644 index 0000000..e10fff4 --- /dev/null +++ b/kubespray/roles/container-engine/kata-containers/molecule/default/tests/test_default.py @@ -0,0 +1,37 @@ +import os + +import testinfra.utils.ansible_runner + +testinfra_hosts = testinfra.utils.ansible_runner.AnsibleRunner( + os.environ['MOLECULE_INVENTORY_FILE']).get_hosts('all') + + +def test_run(host): + kataruntime = "/opt/kata/bin/kata-runtime" + with host.sudo(): + cmd = host.command(kataruntime + " version") + assert cmd.rc == 0 + assert "kata-runtime" in cmd.stdout + + +def test_run_check(host): + kataruntime = "/opt/kata/bin/kata-runtime" + with host.sudo(): + cmd = host.command(kataruntime + " check") + assert cmd.rc == 0 + assert "System is capable of running" in cmd.stdout + + +def test_run_pod(host): + runtime = "kata-qemu" + + run_command = "/usr/local/bin/crictl run --with-pull --runtime {} /tmp/container.json /tmp/sandbox.json".format(runtime) + with host.sudo(): + cmd = host.command(run_command) + assert cmd.rc == 0 + + with host.sudo(): + log_f = host.file("/tmp/kata1.0.log") + + assert log_f.exists + assert b"Hello from Docker" in log_f.content diff --git a/kubespray/roles/container-engine/kata-containers/tasks/main.yml b/kubespray/roles/container-engine/kata-containers/tasks/main.yml new file mode 100644 index 0000000..e795b1f --- /dev/null +++ b/kubespray/roles/container-engine/kata-containers/tasks/main.yml @@ -0,0 +1,54 @@ +--- +- name: Kata-containers | Download kata binary + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.kata_containers) }}" + +- name: Kata-containers | Copy kata-containers binary + unarchive: + src: "{{ downloads.kata_containers.dest }}" + dest: "/" + mode: 0755 + owner: root + group: root + remote_src: yes + +- name: Kata-containers | Create config directory + file: + path: "{{ kata_containers_config_dir }}" + state: directory + mode: 0755 + +- name: Kata-containers | Set configuration + template: + src: "{{ item }}.j2" + dest: "{{ kata_containers_config_dir }}/{{ item }}" + mode: 0644 + with_items: + - configuration-qemu.toml + +- name: Kata-containers | Set containerd bin + vars: + shim: "{{ item }}" + template: + dest: "{{ kata_containers_containerd_bin_dir }}/containerd-shim-kata-{{ item }}-v2" + src: containerd-shim-kata-v2.j2 + mode: 0755 + with_items: + - qemu + +- name: Kata-containers | Load vhost kernel modules + community.general.modprobe: + state: present + name: "{{ item }}" + with_items: + - vhost_vsock + - vhost_net + +- name: Kata-containers | Persist vhost kernel modules + copy: + dest: /etc/modules-load.d/kubespray-kata-containers.conf + mode: 0644 + content: | + vhost_vsock + vhost_net diff --git a/kubespray/roles/container-engine/kata-containers/templates/configuration-qemu.toml.j2 b/kubespray/roles/container-engine/kata-containers/templates/configuration-qemu.toml.j2 new file mode 100644 index 0000000..1551144 --- /dev/null +++ b/kubespray/roles/container-engine/kata-containers/templates/configuration-qemu.toml.j2 @@ -0,0 +1,706 @@ +# Copyright (c) 2017-2019 Intel Corporation +# Copyright (c) 2021 Adobe Inc. +# +# SPDX-License-Identifier: Apache-2.0 +# + +# XXX: WARNING: this file is auto-generated. +# XXX: +# XXX: Source file: "config/configuration-qemu.toml.in" +# XXX: Project: +# XXX: Name: Kata Containers +# XXX: Type: kata + +[hypervisor.qemu] +path = "/opt/kata/bin/qemu-system-x86_64" +{% if kata_containers_version is version('2.2.0', '>=') %} +kernel = "/opt/kata/share/kata-containers/vmlinux.container" +{% else %} +kernel = "/opt/kata/share/kata-containers/vmlinuz.container" +{% endif %} +image = "/opt/kata/share/kata-containers/kata-containers.img" +# initrd = "/opt/kata/share/kata-containers/kata-containers-initrd.img" +machine_type = "q35" + +# rootfs filesystem type: +# - ext4 (default) +# - xfs +# - erofs +rootfs_type="ext4" + +# Enable confidential guest support. +# Toggling that setting may trigger different hardware features, ranging +# from memory encryption to both memory and CPU-state encryption and integrity. +# The Kata Containers runtime dynamically detects the available feature set and +# aims at enabling the largest possible one, returning an error if none is +# available, or none is supported by the hypervisor. +# +# Known limitations: +# * Does not work by design: +# - CPU Hotplug +# - Memory Hotplug +# - NVDIMM devices +# +# Default false +# confidential_guest = true + +# Choose AMD SEV-SNP confidential guests +# In case of using confidential guests on AMD hardware that supports both SEV +# and SEV-SNP, the following enables SEV-SNP guests. SEV guests are default. +# Default false +# sev_snp_guest = true + +# Enable running QEMU VMM as a non-root user. +# By default QEMU VMM run as root. When this is set to true, QEMU VMM process runs as +# a non-root random user. See documentation for the limitations of this mode. +# rootless = true + +# List of valid annotation names for the hypervisor +# Each member of the list is a regular expression, which is the base name +# of the annotation, e.g. "path" for io.katacontainers.config.hypervisor.path" +enable_annotations = ["enable_iommu"] + +# List of valid annotations values for the hypervisor +# Each member of the list is a path pattern as described by glob(3). +# The default if not set is empty (all annotations rejected.) +# Your distribution recommends: ["/opt/kata/bin/qemu-system-x86_64"] +valid_hypervisor_paths = ["/opt/kata/bin/qemu-system-x86_64"] + +# Optional space-separated list of options to pass to the guest kernel. +# For example, use `kernel_params = "vsyscall=emulate"` if you are having +# trouble running pre-2.15 glibc. +# +# WARNING: - any parameter specified here will take priority over the default +# parameter value of the same name used to start the virtual machine. +# Do not set values here unless you understand the impact of doing so as you +# may stop the virtual machine from booting. +# To see the list of default parameters, enable hypervisor debug, create a +# container and look for 'default-kernel-parameters' log entries. +kernel_params = "" + +# Path to the firmware. +# If you want that qemu uses the default firmware leave this option empty +firmware = "" + +# Path to the firmware volume. +# firmware TDVF or OVMF can be split into FIRMWARE_VARS.fd (UEFI variables +# as configuration) and FIRMWARE_CODE.fd (UEFI program image). UEFI variables +# can be customized per each user while UEFI code is kept same. +firmware_volume = "" + +# Machine accelerators +# comma-separated list of machine accelerators to pass to the hypervisor. +# For example, `machine_accelerators = "nosmm,nosmbus,nosata,nopit,static-prt,nofw"` +machine_accelerators="" + +# Qemu seccomp sandbox feature +# comma-separated list of seccomp sandbox features to control the syscall access. +# For example, `seccompsandbox= "on,obsolete=deny,spawn=deny,resourcecontrol=deny"` +# Note: "elevateprivileges=deny" doesn't work with daemonize option, so it's removed from the seccomp sandbox +# Another note: enabling this feature may reduce performance, you may enable +# /proc/sys/net/core/bpf_jit_enable to reduce the impact. see https://man7.org/linux/man-pages/man8/bpfc.8.html +#seccompsandbox="on,obsolete=deny,spawn=deny,resourcecontrol=deny" + +# CPU features +# comma-separated list of cpu features to pass to the cpu +# For example, `cpu_features = "pmu=off,vmx=off" +cpu_features="pmu=off" + +# Default number of vCPUs per SB/VM: +# unspecified or 0 --> will be set to 1 +# < 0 --> will be set to the actual number of physical cores +# > 0 <= number of physical cores --> will be set to the specified number +# > number of physical cores --> will be set to the actual number of physical cores +default_vcpus = 1 + +# Default maximum number of vCPUs per SB/VM: +# unspecified or == 0 --> will be set to the actual number of physical cores or to the maximum number +# of vCPUs supported by KVM if that number is exceeded +# > 0 <= number of physical cores --> will be set to the specified number +# > number of physical cores --> will be set to the actual number of physical cores or to the maximum number +# of vCPUs supported by KVM if that number is exceeded +# WARNING: Depending of the architecture, the maximum number of vCPUs supported by KVM is used when +# the actual number of physical cores is greater than it. +# WARNING: Be aware that this value impacts the virtual machine's memory footprint and CPU +# the hotplug functionality. For example, `default_maxvcpus = 240` specifies that until 240 vCPUs +# can be added to a SB/VM, but the memory footprint will be big. Another example, with +# `default_maxvcpus = 8` the memory footprint will be small, but 8 will be the maximum number of +# vCPUs supported by the SB/VM. In general, we recommend that you do not edit this variable, +# unless you know what are you doing. +# NOTICE: on arm platform with gicv2 interrupt controller, set it to 8. +default_maxvcpus = 0 + +# Bridges can be used to hot plug devices. +# Limitations: +# * Currently only pci bridges are supported +# * Until 30 devices per bridge can be hot plugged. +# * Until 5 PCI bridges can be cold plugged per VM. +# This limitation could be a bug in qemu or in the kernel +# Default number of bridges per SB/VM: +# unspecified or 0 --> will be set to 1 +# > 1 <= 5 --> will be set to the specified number +# > 5 --> will be set to 5 +default_bridges = 1 + +# Default memory size in MiB for SB/VM. +# If unspecified then it will be set 2048 MiB. +default_memory = {{ kata_containers_qemu_default_memory }} +# +# Default memory slots per SB/VM. +# If unspecified then it will be set 10. +# This is will determine the times that memory will be hotadded to sandbox/VM. +#memory_slots = 10 + +# Default maximum memory in MiB per SB / VM +# unspecified or == 0 --> will be set to the actual amount of physical RAM +# > 0 <= amount of physical RAM --> will be set to the specified number +# > amount of physical RAM --> will be set to the actual amount of physical RAM +default_maxmemory = 0 + +# The size in MiB will be plused to max memory of hypervisor. +# It is the memory address space for the NVDIMM devie. +# If set block storage driver (block_device_driver) to "nvdimm", +# should set memory_offset to the size of block device. +# Default 0 +#memory_offset = 0 + +# Specifies virtio-mem will be enabled or not. +# Please note that this option should be used with the command +# "echo 1 > /proc/sys/vm/overcommit_memory". +# Default false +#enable_virtio_mem = true + +# Disable block device from being used for a container's rootfs. +# In case of a storage driver like devicemapper where a container's +# root file system is backed by a block device, the block device is passed +# directly to the hypervisor for performance reasons. +# This flag prevents the block device from being passed to the hypervisor, +# virtio-fs is used instead to pass the rootfs. +disable_block_device_use = false + +# Shared file system type: +# - virtio-fs (default) +# - virtio-9p +# - virtio-fs-nydus +{% if kata_containers_version is version('2.2.0', '>=') %} +shared_fs = "virtio-fs" +{% else %} +shared_fs = "virtio-9p" +{% endif %} + +# Path to vhost-user-fs daemon. +{% if kata_containers_version is version('2.5.0', '>=') %} +virtio_fs_daemon = "/opt/kata/libexec/virtiofsd" +{% else %} +virtio_fs_daemon = "/opt/kata/libexec/kata-qemu/virtiofsd" +{% endif %} + +# List of valid annotations values for the virtiofs daemon +# The default if not set is empty (all annotations rejected.) +# Your distribution recommends: ["/opt/kata/libexec/virtiofsd"] +valid_virtio_fs_daemon_paths = [ + "/opt/kata/libexec/virtiofsd", + "/opt/kata/libexec/kata-qemu/virtiofsd", +] + +# Default size of DAX cache in MiB +virtio_fs_cache_size = 0 + +# Default size of virtqueues +virtio_fs_queue_size = 1024 + +# Extra args for virtiofsd daemon +# +# Format example: +# ["--arg1=xxx", "--arg2=yyy"] +# Examples: +# Set virtiofsd log level to debug : ["--log-level=debug"] +# +# see `virtiofsd -h` for possible options. +virtio_fs_extra_args = ["--thread-pool-size=1", "--announce-submounts"] + +# Cache mode: +# +# - never +# Metadata, data, and pathname lookup are not cached in guest. They are +# always fetched from host and any changes are immediately pushed to host. +# +# - auto +# Metadata and pathname lookup cache expires after a configured amount of +# time (default is 1 second). Data is cached while the file is open (close +# to open consistency). +# +# - always +# Metadata, data, and pathname lookup are cached in guest and never expire. +virtio_fs_cache = "{{ kata_containers_virtio_fs_cache }}" + +# Block storage driver to be used for the hypervisor in case the container +# rootfs is backed by a block device. This is virtio-scsi, virtio-blk +# or nvdimm. +block_device_driver = "virtio-scsi" + +# aio is the I/O mechanism used by qemu +# Options: +# +# - threads +# Pthread based disk I/O. +# +# - native +# Native Linux I/O. +# +# - io_uring +# Linux io_uring API. This provides the fastest I/O operations on Linux, requires kernel>5.1 and +# qemu >=5.0. +block_device_aio = "io_uring" + +# Specifies cache-related options will be set to block devices or not. +# Default false +#block_device_cache_set = true + +# Specifies cache-related options for block devices. +# Denotes whether use of O_DIRECT (bypass the host page cache) is enabled. +# Default false +#block_device_cache_direct = true + +# Specifies cache-related options for block devices. +# Denotes whether flush requests for the device are ignored. +# Default false +#block_device_cache_noflush = true + +# Enable iothreads (data-plane) to be used. This causes IO to be +# handled in a separate IO thread. This is currently only implemented +# for SCSI. +# +enable_iothreads = false + +# Enable pre allocation of VM RAM, default false +# Enabling this will result in lower container density +# as all of the memory will be allocated and locked +# This is useful when you want to reserve all the memory +# upfront or in the cases where you want memory latencies +# to be very predictable +# Default false +enable_mem_prealloc = {{ kata_containers_qemu_enable_mem_prealloc }} + +# Enable huge pages for VM RAM, default false +# Enabling this will result in the VM memory +# being allocated using huge pages. +# This is useful when you want to use vhost-user network +# stacks within the container. This will automatically +# result in memory pre allocation +#enable_hugepages = true + +# Enable vhost-user storage device, default false +# Enabling this will result in some Linux reserved block type +# major range 240-254 being chosen to represent vhost-user devices. +enable_vhost_user_store = false + +# The base directory specifically used for vhost-user devices. +# Its sub-path "block" is used for block devices; "block/sockets" is +# where we expect vhost-user sockets to live; "block/devices" is where +# simulated block device nodes for vhost-user devices to live. +vhost_user_store_path = "/var/run/kata-containers/vhost-user" + +# Enable vIOMMU, default false +# Enabling this will result in the VM having a vIOMMU device +# This will also add the following options to the kernel's +# command line: intel_iommu=on,iommu=pt +#enable_iommu = true + +# Enable IOMMU_PLATFORM, default false +# Enabling this will result in the VM device having iommu_platform=on set +#enable_iommu_platform = true + +# List of valid annotations values for the vhost user store path +# The default if not set is empty (all annotations rejected.) +# Your distribution recommends: ["/var/run/kata-containers/vhost-user"] +valid_vhost_user_store_paths = ["/var/run/kata-containers/vhost-user"] + +# The timeout for reconnecting on non-server spdk sockets when the remote end goes away. +# qemu will delay this many seconds and then attempt to reconnect. +# Zero disables reconnecting, and the default is zero. +vhost_user_reconnect_timeout_sec = 0 + +# Enable file based guest memory support. The default is an empty string which +# will disable this feature. In the case of virtio-fs, this is enabled +# automatically and '/dev/shm' is used as the backing folder. +# This option will be ignored if VM templating is enabled. +#file_mem_backend = "" + +# List of valid annotations values for the file_mem_backend annotation +# The default if not set is empty (all annotations rejected.) +# Your distribution recommends: [""] +valid_file_mem_backends = [""] + +# -pflash can add image file to VM. The arguments of it should be in format +# of ["/path/to/flash0.img", "/path/to/flash1.img"] +pflashes = [] + +# This option changes the default hypervisor and kernel parameters +# to enable debug output where available. And Debug also enables the hmp socket. +# +# Default false +enable_debug = {{ kata_containers_qemu_debug }} + +# Disable the customizations done in the runtime when it detects +# that it is running on top a VMM. This will result in the runtime +# behaving as it would when running on bare metal. +# +#disable_nesting_checks = true + +# This is the msize used for 9p shares. It is the number of bytes +# used for 9p packet payload. +#msize_9p = 8192 + +# If false and nvdimm is supported, use nvdimm device to plug guest image. +# Otherwise virtio-block device is used. +# +# nvdimm is not supported when `confidential_guest = true`. +# +# Default is false +#disable_image_nvdimm = true + +# VFIO devices are hotplugged on a bridge by default. +# Enable hotplugging on root bus. This may be required for devices with +# a large PCI bar, as this is a current limitation with hotplugging on +# a bridge. +# Default false +#hotplug_vfio_on_root_bus = true + +# Before hot plugging a PCIe device, you need to add a pcie_root_port device. +# Use this parameter when using some large PCI bar devices, such as Nvidia GPU +# The value means the number of pcie_root_port +# This value is valid when hotplug_vfio_on_root_bus is true and machine_type is "q35" +# Default 0 +#pcie_root_port = 2 + +# If vhost-net backend for virtio-net is not desired, set to true. Default is false, which trades off +# security (vhost-net runs ring0) for network I/O performance. +#disable_vhost_net = true + +# +# Default entropy source. +# The path to a host source of entropy (including a real hardware RNG) +# /dev/urandom and /dev/random are two main options. +# Be aware that /dev/random is a blocking source of entropy. If the host +# runs out of entropy, the VMs boot time will increase leading to get startup +# timeouts. +# The source of entropy /dev/urandom is non-blocking and provides a +# generally acceptable source of entropy. It should work well for pretty much +# all practical purposes. +#entropy_source= "/dev/urandom" + +# List of valid annotations values for entropy_source +# The default if not set is empty (all annotations rejected.) +# Your distribution recommends: ["/dev/urandom","/dev/random",""] +valid_entropy_sources = ["/dev/urandom","/dev/random",""] + +# Path to OCI hook binaries in the *guest rootfs*. +# This does not affect host-side hooks which must instead be added to +# the OCI spec passed to the runtime. +# +# You can create a rootfs with hooks by customizing the osbuilder scripts: +# https://github.com/kata-containers/kata-containers/tree/main/tools/osbuilder +# +# Hooks must be stored in a subdirectory of guest_hook_path according to their +# hook type, i.e. "guest_hook_path/{prestart,poststart,poststop}". +# The agent will scan these directories for executable files and add them, in +# lexicographical order, to the lifecycle of the guest container. +# Hooks are executed in the runtime namespace of the guest. See the official documentation: +# https://github.com/opencontainers/runtime-spec/blob/v1.0.1/config.md#posix-platform-hooks +# Warnings will be logged if any error is encountered while scanning for hooks, +# but it will not abort container execution. +#guest_hook_path = "/usr/share/oci/hooks" +# +# Use rx Rate Limiter to control network I/O inbound bandwidth(size in bits/sec for SB/VM). +# In Qemu, we use classful qdiscs HTB(Hierarchy Token Bucket) to discipline traffic. +# Default 0-sized value means unlimited rate. +#rx_rate_limiter_max_rate = 0 +# Use tx Rate Limiter to control network I/O outbound bandwidth(size in bits/sec for SB/VM). +# In Qemu, we use classful qdiscs HTB(Hierarchy Token Bucket) and ifb(Intermediate Functional Block) +# to discipline traffic. +# Default 0-sized value means unlimited rate. +#tx_rate_limiter_max_rate = 0 + +# Set where to save the guest memory dump file. +# If set, when GUEST_PANICKED event occurred, +# guest memeory will be dumped to host filesystem under guest_memory_dump_path, +# This directory will be created automatically if it does not exist. +# +# The dumped file(also called vmcore) can be processed with crash or gdb. +# +# WARNING: +# Dump guest’s memory can take very long depending on the amount of guest memory +# and use much disk space. +#guest_memory_dump_path="/var/crash/kata" + +# If enable paging. +# Basically, if you want to use "gdb" rather than "crash", +# or need the guest-virtual addresses in the ELF vmcore, +# then you should enable paging. +# +# See: https://www.qemu.org/docs/master/qemu-qmp-ref.html#Dump-guest-memory for details +#guest_memory_dump_paging=false + +# Enable swap in the guest. Default false. +# When enable_guest_swap is enabled, insert a raw file to the guest as the swap device +# if the swappiness of a container (set by annotation "io.katacontainers.container.resource.swappiness") +# is bigger than 0. +# The size of the swap device should be +# swap_in_bytes (set by annotation "io.katacontainers.container.resource.swap_in_bytes") - memory_limit_in_bytes. +# If swap_in_bytes is not set, the size should be memory_limit_in_bytes. +# If swap_in_bytes and memory_limit_in_bytes is not set, the size should +# be default_memory. +#enable_guest_swap = true + +# use legacy serial for guest console if available and implemented for architecture. Default false +#use_legacy_serial = true + +# disable applying SELinux on the VMM process (default false) +disable_selinux=false + +# disable applying SELinux on the container process +# If set to false, the type `container_t` is applied to the container process by default. +# Note: To enable guest SELinux, the guest rootfs must be CentOS that is created and built +# with `SELINUX=yes`. +# (default: true) +disable_guest_selinux=true + +[factory] +# VM templating support. Once enabled, new VMs are created from template +# using vm cloning. They will share the same initial kernel, initramfs and +# agent memory by mapping it readonly. It helps speeding up new container +# creation and saves a lot of memory if there are many kata containers running +# on the same host. +# +# When disabled, new VMs are created from scratch. +# +# Note: Requires "initrd=" to be set ("image=" is not supported). +# +# Default false +#enable_template = true + +# Specifies the path of template. +# +# Default "/run/vc/vm/template" +#template_path = "/run/vc/vm/template" + +# The number of caches of VMCache: +# unspecified or == 0 --> VMCache is disabled +# > 0 --> will be set to the specified number +# +# VMCache is a function that creates VMs as caches before using it. +# It helps speed up new container creation. +# The function consists of a server and some clients communicating +# through Unix socket. The protocol is gRPC in protocols/cache/cache.proto. +# The VMCache server will create some VMs and cache them by factory cache. +# It will convert the VM to gRPC format and transport it when gets +# requestion from clients. +# Factory grpccache is the VMCache client. It will request gRPC format +# VM and convert it back to a VM. If VMCache function is enabled, +# kata-runtime will request VM from factory grpccache when it creates +# a new sandbox. +# +# Default 0 +#vm_cache_number = 0 + +# Specify the address of the Unix socket that is used by VMCache. +# +# Default /var/run/kata-containers/cache.sock +#vm_cache_endpoint = "/var/run/kata-containers/cache.sock" + +[agent.kata] +# If enabled, make the agent display debug-level messages. +# (default: disabled) +enable_debug = {{ kata_containers_qemu_debug }} + +# Enable agent tracing. +# +# If enabled, the agent will generate OpenTelemetry trace spans. +# +# Notes: +# +# - If the runtime also has tracing enabled, the agent spans will be +# associated with the appropriate runtime parent span. +# - If enabled, the runtime will wait for the container to shutdown, +# increasing the container shutdown time slightly. +# +# (default: disabled) +#enable_tracing = true + +# Comma separated list of kernel modules and their parameters. +# These modules will be loaded in the guest kernel using modprobe(8). +# The following example can be used to load two kernel modules with parameters +# - kernel_modules=["e1000e InterruptThrottleRate=3000,3000,3000 EEE=1", "i915 enable_ppgtt=0"] +# The first word is considered as the module name and the rest as its parameters. +# Container will not be started when: +# * A kernel module is specified and the modprobe command is not installed in the guest +# or it fails loading the module. +# * The module is not available in the guest or it doesn't met the guest kernel +# requirements, like architecture and version. +# +kernel_modules=[] + +# Enable debug console. + +# If enabled, user can connect guest OS running inside hypervisor +# through "kata-runtime exec " command + +#debug_console_enabled = true + +# Agent connection dialing timeout value in seconds +# (default: 30) +#dial_timeout = 30 + +[runtime] +# If enabled, the runtime will log additional debug messages to the +# system log +# (default: disabled) +enable_debug = {{ kata_containers_qemu_debug }} +# +# Internetworking model +# Determines how the VM should be connected to the +# the container network interface +# Options: +# +# - macvtap +# Used when the Container network interface can be bridged using +# macvtap. +# +# - none +# Used when customize network. Only creates a tap device. No veth pair. +# +# - tcfilter +# Uses tc filter rules to redirect traffic from the network interface +# provided by plugin to a tap interface connected to the VM. +# +internetworking_model="tcfilter" + +# disable guest seccomp +# Determines whether container seccomp profiles are passed to the virtual +# machine and applied by the kata agent. If set to true, seccomp is not applied +# within the guest +# (default: true) +disable_guest_seccomp=true + +# vCPUs pinning settings +# if enabled, each vCPU thread will be scheduled to a fixed CPU +# qualified condition: num(vCPU threads) == num(CPUs in sandbox's CPUSet) +# enable_vcpus_pinning = false + +# Apply a custom SELinux security policy to the container process inside the VM. +# This is used when you want to apply a type other than the default `container_t`, +# so general users should not uncomment and apply it. +# (format: "user:role:type") +# Note: You cannot specify MCS policy with the label because the sensitivity levels and +# categories are determined automatically by high-level container runtimes such as containerd. +#guest_selinux_label="system_u:system_r:container_t" + +# If enabled, the runtime will create opentracing.io traces and spans. +# (See https://www.jaegertracing.io/docs/getting-started). +# (default: disabled) +#enable_tracing = true + +# Set the full url to the Jaeger HTTP Thrift collector. +# The default if not set will be "http://localhost:14268/api/traces" +#jaeger_endpoint = "" + +# Sets the username to be used if basic auth is required for Jaeger. +#jaeger_user = "" + +# Sets the password to be used if basic auth is required for Jaeger. +#jaeger_password = "" + +# If enabled, the runtime will not create a network namespace for shim and hypervisor processes. +# This option may have some potential impacts to your host. It should only be used when you know what you're doing. +# `disable_new_netns` conflicts with `internetworking_model=tcfilter` and `internetworking_model=macvtap`. It works only +# with `internetworking_model=none`. The tap device will be in the host network namespace and can connect to a bridge +# (like OVS) directly. +# (default: false) +#disable_new_netns = true + +# if enabled, the runtime will add all the kata processes inside one dedicated cgroup. +# The container cgroups in the host are not created, just one single cgroup per sandbox. +# The runtime caller is free to restrict or collect cgroup stats of the overall Kata sandbox. +# The sandbox cgroup path is the parent cgroup of a container with the PodSandbox annotation. +# The sandbox cgroup is constrained if there is no container type annotation. +# See: https://pkg.go.dev/github.com/kata-containers/kata-containers/src/runtime/virtcontainers#ContainerType +sandbox_cgroup_only={{ kata_containers_qemu_sandbox_cgroup_only }} + +# If enabled, the runtime will attempt to determine appropriate sandbox size (memory, CPU) before booting the virtual machine. In +# this case, the runtime will not dynamically update the amount of memory and CPU in the virtual machine. This is generally helpful +# when a hardware architecture or hypervisor solutions is utilized which does not support CPU and/or memory hotplug. +# Compatibility for determining appropriate sandbox (VM) size: +# - When running with pods, sandbox sizing information will only be available if using Kubernetes >= 1.23 and containerd >= 1.6. CRI-O +# does not yet support sandbox sizing annotations. +# - When running single containers using a tool like ctr, container sizing information will be available. +static_sandbox_resource_mgmt=false + +# If specified, sandbox_bind_mounts identifieds host paths to be mounted (ro) into the sandboxes shared path. +# This is only valid if filesystem sharing is utilized. The provided path(s) will be bindmounted into the shared fs directory. +# If defaults are utilized, these mounts should be available in the guest at `/run/kata-containers/shared/containers/sandbox-mounts` +# These will not be exposed to the container workloads, and are only provided for potential guest services. +sandbox_bind_mounts=[] + +# VFIO Mode +# Determines how VFIO devices should be be presented to the container. +# Options: +# +# - vfio +# Matches behaviour of OCI runtimes (e.g. runc) as much as +# possible. VFIO devices will appear in the container as VFIO +# character devices under /dev/vfio. The exact names may differ +# from the host (they need to match the VM's IOMMU group numbers +# rather than the host's) +# +# - guest-kernel +# This is a Kata-specific behaviour that's useful in certain cases. +# The VFIO device is managed by whatever driver in the VM kernel +# claims it. This means it will appear as one or more device nodes +# or network interfaces depending on the nature of the device. +# Using this mode requires specially built workloads that know how +# to locate the relevant device interfaces within the VM. +# +vfio_mode="guest-kernel" + +# If enabled, the runtime will not create Kubernetes emptyDir mounts on the guest filesystem. Instead, emptyDir mounts will +# be created on the host and shared via virtio-fs. This is potentially slower, but allows sharing of files from host to guest. +disable_guest_empty_dir=false + +# Enabled experimental feature list, format: ["a", "b"]. +# Experimental features are features not stable enough for production, +# they may break compatibility, and are prepared for a big version bump. +# Supported experimental features: +# (default: []) +experimental=[] + +# If enabled, user can run pprof tools with shim v2 process through kata-monitor. +# (default: false) +# enable_pprof = true + +# WARNING: All the options in the following section have not been implemented yet. +# This section was added as a placeholder. DO NOT USE IT! +[image] +# Container image service. +# +# Offload the CRI image management service to the Kata agent. +# (default: false) +#service_offload = true + +# Container image decryption keys provisioning. +# Applies only if service_offload is true. +# Keys can be provisioned locally (e.g. through a special command or +# a local file) or remotely (usually after the guest is remotely attested). +# The provision setting is a complete URL that lets the Kata agent decide +# which method to use in order to fetch the keys. +# +# Keys can be stored in a local file, in a measured and attested initrd: +#provision=data:///local/key/file +# +# Keys could be fetched through a special command or binary from the +# initrd (guest) image, e.g. a firmware call: +#provision=file:///path/to/bin/fetcher/in/guest +# +# Keys can be remotely provisioned. The Kata agent fetches them from e.g. +# a HTTPS URL: +#provision=https://my-key-broker.foo/tenant/ diff --git a/kubespray/roles/container-engine/kata-containers/templates/containerd-shim-kata-v2.j2 b/kubespray/roles/container-engine/kata-containers/templates/containerd-shim-kata-v2.j2 new file mode 100644 index 0000000..a3cb830 --- /dev/null +++ b/kubespray/roles/container-engine/kata-containers/templates/containerd-shim-kata-v2.j2 @@ -0,0 +1,2 @@ +#!/bin/bash +KATA_CONF_FILE={{ kata_containers_config_dir }}/configuration-{{ shim }}.toml {{ kata_containers_dir }}/bin/containerd-shim-kata-v2 $@ diff --git a/kubespray/roles/container-engine/meta/main.yml b/kubespray/roles/container-engine/meta/main.yml new file mode 100644 index 0000000..3e068d6 --- /dev/null +++ b/kubespray/roles/container-engine/meta/main.yml @@ -0,0 +1,58 @@ +# noqa role-name - this is a meta role that doesn't need a name +--- +dependencies: + - role: container-engine/validate-container-engine + tags: + - container-engine + - validate-container-engine + + - role: container-engine/kata-containers + when: + - kata_containers_enabled + tags: + - container-engine + - kata-containers + + - role: container-engine/gvisor + when: + - gvisor_enabled + - container_manager in ['docker', 'containerd'] + tags: + - container-engine + - gvisor + + - role: container-engine/crun + when: + - crun_enabled + tags: + - container-engine + - crun + + - role: container-engine/youki + when: + - youki_enabled + - container_manager == 'crio' + tags: + - container-engine + - youki + + - role: container-engine/cri-o + when: + - container_manager == 'crio' + tags: + - container-engine + - crio + + - role: container-engine/containerd + when: + - container_manager == 'containerd' + tags: + - container-engine + - containerd + + - role: container-engine/cri-dockerd + when: + - container_manager == 'docker' + tags: + - container-engine + - docker diff --git a/kubespray/roles/container-engine/nerdctl/handlers/main.yml b/kubespray/roles/container-engine/nerdctl/handlers/main.yml new file mode 100644 index 0000000..27895ff --- /dev/null +++ b/kubespray/roles/container-engine/nerdctl/handlers/main.yml @@ -0,0 +1,12 @@ +--- +- name: Get nerdctl completion + command: "{{ bin_dir }}/nerdctl completion bash" + changed_when: False + register: nerdctl_completion + check_mode: false + +- name: Install nerdctl completion + copy: + dest: /etc/bash_completion.d/nerdctl + content: "{{ nerdctl_completion.stdout }}" + mode: 0644 diff --git a/kubespray/roles/container-engine/nerdctl/tasks/main.yml b/kubespray/roles/container-engine/nerdctl/tasks/main.yml new file mode 100644 index 0000000..e4e4ebd --- /dev/null +++ b/kubespray/roles/container-engine/nerdctl/tasks/main.yml @@ -0,0 +1,36 @@ +--- +- name: Nerdctl | Download nerdctl + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.nerdctl) }}" + +- name: Nerdctl | Copy nerdctl binary from download dir + copy: + src: "{{ local_release_dir }}/nerdctl" + dest: "{{ bin_dir }}/nerdctl" + mode: 0755 + remote_src: true + owner: root + group: root + become: true + notify: + - Get nerdctl completion + - Install nerdctl completion + +- name: Nerdctl | Create configuration dir + file: + path: /etc/nerdctl + state: directory + mode: 0755 + owner: root + group: root + become: true + +- name: Nerdctl | Install nerdctl configuration + template: + src: nerdctl.toml.j2 + dest: /etc/nerdctl/nerdctl.toml + mode: 0644 + owner: root + group: root + become: true diff --git a/kubespray/roles/container-engine/nerdctl/templates/nerdctl.toml.j2 b/kubespray/roles/container-engine/nerdctl/templates/nerdctl.toml.j2 new file mode 100644 index 0000000..8b590f6 --- /dev/null +++ b/kubespray/roles/container-engine/nerdctl/templates/nerdctl.toml.j2 @@ -0,0 +1,9 @@ +debug = false +debug_full = false +address = "{{ cri_socket }}" +namespace = "k8s.io" +snapshotter = "{{ nerdctl_snapshotter | default('overlayfs') }}" +cni_path = "/opt/cni/bin" +cni_netconfpath = "/etc/cni/net.d" +cgroup_manager = "{{ kubelet_cgroup_driver | default('systemd') }}" +hosts_dir = ["{{ containerd_cfg_dir }}/certs.d"] diff --git a/kubespray/roles/container-engine/runc/defaults/main.yml b/kubespray/roles/container-engine/runc/defaults/main.yml new file mode 100644 index 0000000..af8aa08 --- /dev/null +++ b/kubespray/roles/container-engine/runc/defaults/main.yml @@ -0,0 +1,5 @@ +--- + +runc_bin_dir: "{{ bin_dir }}" + +runc_package_name: runc diff --git a/kubespray/roles/container-engine/runc/tasks/main.yml b/kubespray/roles/container-engine/runc/tasks/main.yml new file mode 100644 index 0000000..542a447 --- /dev/null +++ b/kubespray/roles/container-engine/runc/tasks/main.yml @@ -0,0 +1,38 @@ +--- +- name: Runc | check if fedora coreos + stat: + path: /run/ostree-booted + get_attributes: no + get_checksum: no + get_mime: no + register: ostree + +- name: Runc | set is_ostree + set_fact: + is_ostree: "{{ ostree.stat.exists }}" + +- name: Runc | Uninstall runc package managed by package manager + package: + name: "{{ runc_package_name }}" + state: absent + when: + - not (is_ostree or (ansible_distribution == "Flatcar Container Linux by Kinvolk") or (ansible_distribution == "Flatcar")) + +- name: Runc | Download runc binary + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.runc) }}" + +- name: Copy runc binary from download dir + copy: + src: "{{ downloads.runc.dest }}" + dest: "{{ runc_bin_dir }}/runc" + mode: 0755 + remote_src: true + +- name: Runc | Remove orphaned binary + file: + path: /usr/bin/runc + state: absent + when: runc_bin_dir != "/usr/bin" + ignore_errors: true # noqa ignore-errors diff --git a/kubespray/roles/container-engine/skopeo/tasks/main.yml b/kubespray/roles/container-engine/skopeo/tasks/main.yml new file mode 100644 index 0000000..cef0424 --- /dev/null +++ b/kubespray/roles/container-engine/skopeo/tasks/main.yml @@ -0,0 +1,32 @@ +--- +- name: Skopeo | check if fedora coreos + stat: + path: /run/ostree-booted + get_attributes: no + get_checksum: no + get_mime: no + register: ostree + +- name: Skopeo | set is_ostree + set_fact: + is_ostree: "{{ ostree.stat.exists }}" + +- name: Skopeo | Uninstall skopeo package managed by package manager + package: + name: skopeo + state: absent + when: + - not (is_ostree or (ansible_distribution == "Flatcar Container Linux by Kinvolk") or (ansible_distribution == "Flatcar")) + ignore_errors: true # noqa ignore-errors + +- name: Skopeo | Download skopeo binary + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.skopeo) }}" + +- name: Copy skopeo binary from download dir + copy: + src: "{{ downloads.skopeo.dest }}" + dest: "{{ bin_dir }}/skopeo" + mode: 0755 + remote_src: true diff --git a/kubespray/roles/container-engine/validate-container-engine/tasks/main.yml b/kubespray/roles/container-engine/validate-container-engine/tasks/main.yml new file mode 100644 index 0000000..08ea1e5 --- /dev/null +++ b/kubespray/roles/container-engine/validate-container-engine/tasks/main.yml @@ -0,0 +1,153 @@ +--- +- name: Validate-container-engine | check if fedora coreos + stat: + path: /run/ostree-booted + get_attributes: no + get_checksum: no + get_mime: no + register: ostree + tags: + - facts + +- name: Validate-container-engine | set is_ostree + set_fact: + is_ostree: "{{ ostree.stat.exists }}" + tags: + - facts + +- name: Ensure kubelet systemd unit exists + stat: + path: "/etc/systemd/system/kubelet.service" + register: kubelet_systemd_unit_exists + tags: + - facts + +- name: Populate service facts + service_facts: + tags: + - facts + +- name: Check if containerd is installed + find: + file_type: file + recurse: yes + use_regex: yes + patterns: + - containerd.service$ + paths: + - /lib/systemd + - /etc/systemd + - /run/systemd + register: containerd_installed + tags: + - facts + +- name: Check if docker is installed + find: + file_type: file + recurse: yes + use_regex: yes + patterns: + - docker.service$ + paths: + - /lib/systemd + - /etc/systemd + - /run/systemd + register: docker_installed + tags: + - facts + +- name: Check if crio is installed + find: + file_type: file + recurse: yes + use_regex: yes + patterns: + - crio.service$ + paths: + - /lib/systemd + - /etc/systemd + - /run/systemd + register: crio_installed + tags: + - facts + +- name: Uninstall containerd + vars: + service_name: containerd.service + when: + - not (is_ostree or (ansible_distribution == "Flatcar Container Linux by Kinvolk") or (ansible_distribution == "Flatcar")) + - container_manager != "containerd" + - docker_installed.matched == 0 + - containerd_installed.matched > 0 + - ansible_facts.services[service_name]['state'] == 'running' + block: + - name: Drain node + include_role: + name: remove-node/pre-remove + apply: + tags: + - pre-remove + when: kubelet_systemd_unit_exists.stat.exists + - name: Stop kubelet + service: + name: kubelet + state: stopped + when: kubelet_systemd_unit_exists.stat.exists + - name: Remove Containerd + import_role: + name: container-engine/containerd + tasks_from: reset + handlers_from: reset + +- name: Uninstall docker + vars: + service_name: docker.service + when: + - not (is_ostree or (ansible_distribution == "Flatcar Container Linux by Kinvolk") or (ansible_distribution == "Flatcar")) + - container_manager != "docker" + - docker_installed.matched > 0 + - ansible_facts.services[service_name]['state'] == 'running' + block: + - name: Drain node + include_role: + name: remove-node/pre-remove + apply: + tags: + - pre-remove + when: kubelet_systemd_unit_exists.stat.exists + - name: Stop kubelet + service: + name: kubelet + state: stopped + when: kubelet_systemd_unit_exists.stat.exists + - name: Remove Docker + import_role: + name: container-engine/docker + tasks_from: reset + +- name: Uninstall crio + vars: + service_name: crio.service + when: + - not (is_ostree or (ansible_distribution == "Flatcar Container Linux by Kinvolk") or (ansible_distribution == "Flatcar")) + - container_manager != "crio" + - crio_installed.matched > 0 + - ansible_facts.services[service_name]['state'] == 'running' + block: + - name: Drain node + include_role: + name: remove-node/pre-remove + apply: + tags: + - pre-remove + when: kubelet_systemd_unit_exists.stat.exists + - name: Stop kubelet + service: + name: kubelet + state: stopped + when: kubelet_systemd_unit_exists.stat.exists + - name: Remove CRI-O + import_role: + name: container-engine/cri-o + tasks_from: reset diff --git a/kubespray/roles/container-engine/youki/defaults/main.yml b/kubespray/roles/container-engine/youki/defaults/main.yml new file mode 100644 index 0000000..2250f22 --- /dev/null +++ b/kubespray/roles/container-engine/youki/defaults/main.yml @@ -0,0 +1,3 @@ +--- + +youki_bin_dir: "{{ bin_dir }}" diff --git a/kubespray/roles/container-engine/youki/molecule/default/converge.yml b/kubespray/roles/container-engine/youki/molecule/default/converge.yml new file mode 100644 index 0000000..11ef8f6 --- /dev/null +++ b/kubespray/roles/container-engine/youki/molecule/default/converge.yml @@ -0,0 +1,11 @@ +--- +- name: Converge + hosts: all + become: true + vars: + youki_enabled: true + container_manager: crio + roles: + - role: kubespray-defaults + - role: container-engine/cri-o + - role: container-engine/youki diff --git a/kubespray/roles/container-engine/youki/molecule/default/files/10-mynet.conf b/kubespray/roles/container-engine/youki/molecule/default/files/10-mynet.conf new file mode 100644 index 0000000..b9fa3ba --- /dev/null +++ b/kubespray/roles/container-engine/youki/molecule/default/files/10-mynet.conf @@ -0,0 +1,17 @@ +{ + "cniVersion": "0.4.0", + "name": "mynet", + "type": "bridge", + "bridge": "cni0", + "isGateway": true, + "ipMasq": true, + "ipam": { + "type": "host-local", + "subnet": "172.19.0.0/24", + "routes": [ + { + "dst": "0.0.0.0/0" + } + ] + } +} diff --git a/kubespray/roles/container-engine/youki/molecule/default/files/container.json b/kubespray/roles/container-engine/youki/molecule/default/files/container.json new file mode 100644 index 0000000..a5d5094 --- /dev/null +++ b/kubespray/roles/container-engine/youki/molecule/default/files/container.json @@ -0,0 +1,10 @@ +{ + "metadata": { + "name": "youki1" + }, + "image": { + "image": "quay.io/kubespray/hello-world:latest" + }, + "log_path": "youki1.0.log", + "linux": {} +} diff --git a/kubespray/roles/container-engine/youki/molecule/default/files/sandbox.json b/kubespray/roles/container-engine/youki/molecule/default/files/sandbox.json new file mode 100644 index 0000000..b2a4ffe --- /dev/null +++ b/kubespray/roles/container-engine/youki/molecule/default/files/sandbox.json @@ -0,0 +1,10 @@ +{ + "metadata": { + "name": "youki1", + "namespace": "default", + "attempt": 1, + "uid": "hdishd83djaidwnduwk28bcsb" + }, + "linux": {}, + "log_directory": "/tmp" +} diff --git a/kubespray/roles/container-engine/youki/molecule/default/molecule.yml b/kubespray/roles/container-engine/youki/molecule/default/molecule.yml new file mode 100644 index 0000000..9ba1927 --- /dev/null +++ b/kubespray/roles/container-engine/youki/molecule/default/molecule.yml @@ -0,0 +1,39 @@ +--- +role_name_check: 1 +driver: + name: vagrant + provider: + name: libvirt +platforms: + - name: ubuntu20 + box: generic/ubuntu2004 + cpus: 1 + memory: 1024 + nested: true + groups: + - kube_control_plane + provider_options: + driver: kvm + - name: almalinux8 + box: almalinux/8 + cpus: 1 + memory: 1024 + nested: true + groups: + - kube_control_plane + provider_options: + driver: kvm +provisioner: + name: ansible + env: + ANSIBLE_ROLES_PATH: ../../../../ + config_options: + defaults: + callbacks_enabled: profile_tasks + timeout: 120 + inventory: + group_vars: + all: + become: true +verifier: + name: testinfra diff --git a/kubespray/roles/container-engine/youki/molecule/default/prepare.yml b/kubespray/roles/container-engine/youki/molecule/default/prepare.yml new file mode 100644 index 0000000..119f58a --- /dev/null +++ b/kubespray/roles/container-engine/youki/molecule/default/prepare.yml @@ -0,0 +1,49 @@ +--- +- name: Prepare generic + hosts: all + become: true + roles: + - role: kubespray-defaults + - role: bootstrap-os + - role: adduser + user: "{{ addusers.kube }}" + tasks: + - name: Download CNI + include_tasks: "../../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.cni) }}" + +- name: Prepare container runtime + hosts: all + become: true + vars: + container_manager: crio + kube_network_plugin: cni + roles: + - role: kubespray-defaults + - role: network_plugin/cni + - role: container-engine/crictl + tasks: + - name: Copy test container files + copy: + src: "{{ item }}" + dest: "/tmp/{{ item }}" + owner: root + mode: 0644 + with_items: + - container.json + - sandbox.json + - name: Create /etc/cni/net.d directory + file: + path: /etc/cni/net.d + state: directory + owner: root + mode: 0755 + - name: Setup CNI + copy: + src: "{{ item }}" + dest: "/etc/cni/net.d/{{ item }}" + owner: root + mode: 0644 + with_items: + - 10-mynet.conf diff --git a/kubespray/roles/container-engine/youki/molecule/default/tests/test_default.py b/kubespray/roles/container-engine/youki/molecule/default/tests/test_default.py new file mode 100644 index 0000000..54ed5c5 --- /dev/null +++ b/kubespray/roles/container-engine/youki/molecule/default/tests/test_default.py @@ -0,0 +1,29 @@ +import os + +import testinfra.utils.ansible_runner + +testinfra_hosts = testinfra.utils.ansible_runner.AnsibleRunner( + os.environ['MOLECULE_INVENTORY_FILE']).get_hosts('all') + + +def test_run(host): + youkiruntime = "/usr/local/bin/youki" + with host.sudo(): + cmd = host.command(youkiruntime + " --version") + assert cmd.rc == 0 + assert "youki" in cmd.stdout + + +def test_run_pod(host): + runtime = "youki" + + run_command = "/usr/local/bin/crictl run --with-pull --runtime {} /tmp/container.json /tmp/sandbox.json".format(runtime) + with host.sudo(): + cmd = host.command(run_command) + assert cmd.rc == 0 + + with host.sudo(): + log_f = host.file("/tmp/youki1.0.log") + + assert log_f.exists + assert b"Hello from Docker" in log_f.content diff --git a/kubespray/roles/container-engine/youki/tasks/main.yml b/kubespray/roles/container-engine/youki/tasks/main.yml new file mode 100644 index 0000000..e88f663 --- /dev/null +++ b/kubespray/roles/container-engine/youki/tasks/main.yml @@ -0,0 +1,12 @@ +--- +- name: Youki | Download youki + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.youki) }}" + +- name: Youki | Copy youki binary from download dir + copy: + src: "{{ local_release_dir }}/youki_{{ youki_version | regex_replace('\\.', '_') }}_linux/youki-{{ youki_version }}/youki" + dest: "{{ youki_bin_dir }}/youki" + mode: 0755 + remote_src: true diff --git a/kubespray/roles/download/defaults/main/checksums.yml b/kubespray/roles/download/defaults/main/checksums.yml new file mode 100644 index 0000000..fd270c2 --- /dev/null +++ b/kubespray/roles/download/defaults/main/checksums.yml @@ -0,0 +1,1226 @@ +--- +crictl_checksums: + arm: + v1.28.0: 1ea267f3872f4b7f311963ab43ce6653ceeaf8727206c889b56587c95497e9dd + v1.27.1: ec24fb7e4d45b7f3f3df254b22333839f9bdbde585187a51c93d695abefbf147 + v1.27.0: 0b6983195cc62bfc98de1f3fc2ee297a7274fb79ccabf413b8a20765f12d522a + v1.26.1: f6b537fd74aed9ccb38be2f49dc9a18859dffb04ed73aba796d3265a1bdb3c57 + v1.26.0: 88891ee29eab097ab1ed88d55094e7bf464f3347bc9f056140e45efeddd15b33 + arm64: + v1.28.0: 06e9224e42bc5e23085751e93cccdac89f7930ba6f7a45b8f8fc70ef663c37c4 + v1.27.1: 322bf64d12f9e5cd9540987d47446bf9b0545ceb1900ef93376418083ad88241 + v1.27.0: 9317560069ded8e7bf8b9488fdb110d9e62f0fbc0e33ed09fe972768b47752bd + v1.26.1: cfa28be524b5da1a6dded455bb497dfead27b1fd089e1161eb008909509be585 + v1.26.0: b632ca705a98edc8ad7806f4279feaff956ac83aa109bba8a85ed81e6b900599 + amd64: + v1.28.0: 8dc78774f7cbeaf787994d386eec663f0a3cf24de1ea4893598096cb39ef2508 + v1.27.1: b70e8d7bde8ec6ab77c737b6c69be8cb518ce446365734c6db95f15c74a93ce8 + v1.27.0: d335d6e16c309fbc3ff1a29a7e49bb253b5c9b4b030990bf7c6b48687f985cee + v1.26.1: 0c1a0f9900c15ee7a55e757bcdc220faca5dd2e1cfc120459ad1f04f08598127 + v1.26.0: cda5e2143bf19f6b548110ffba0fe3565e03e8743fadd625fee3d62fc4134eed + ppc64le: + v1.28.0: b70fb7bee5982aa1318ba25088319f1d0d1415567f1f76cd69011b8a14da4daf + v1.27.1: c408bb5e797bf02215acf9604c43007bd09cf69353cefa8f20f2c16ab1728a85 + v1.27.0: 3e4301c2d4b561d861970004002fe15d49af907963de06c70d326f2af1f145e0 + v1.26.1: e3026d88722b40deec87711c897df99db3585e2caea17ebd79df5c78f9296583 + v1.26.0: 5538c88b8ccde419e6158ab9c06dfcca1fa0abecf33d0a75b2d22ceddd283f0d +crio_archive_checksums: + arm: + v1.28.1: 0 + v1.28.0: 0 + v1.27.1: 0 + v1.27.0: 0 + v1.26.4: 0 + v1.26.3: 0 + v1.26.2: 0 + v1.26.1: 0 + v1.26.0: 0 + arm64: + v1.28.1: 98a96c6b6bdf20c60e1a7948847c28b57d9e6e47e396b2e405811ea2c24ab9dc + v1.28.0: c8ea800244d9e4ce74af85126afadea2939cd6f7ddd152d0f09fafbf294ef1cc + v1.27.1: ddf601e28dc22d878cdd34549402a236afaa47e0a08f39b09e65bab7034b1b97 + v1.27.0: c6615360311bff7fdfe1933e8d5030a2e9926b7196c4e7a07fcb10e51a676272 + v1.26.4: dbc64d796eb9055f2e070476bb1f32ab7b7bf42ef0ec23212c51beabfd5ac43f + v1.26.3: c85ea3f6476b354af0b15ad7ab80ae202d082ed0c83f1a323b48352c4698db9a + v1.26.2: 8bd9c912de7f8805c162e089a34ca29e607c48a149940193466ccf7bdf74f606 + v1.26.1: 30fe91a60c54b627962da0c21f947424d3cdf484067bc5cda3b3777c10c85384 + v1.26.0: 8605b166d00c674e6363ee2336600fa6c6730a6a724f03ab3b72a0d5f9efcd1d + amd64: + v1.28.1: 63cee2e67e283e29d790caa52531bcca7bc59473fb73bde75f4fd8daa169d4bf + v1.28.0: fa87497c12815766d18f332b38a4d823fa6ad6bb3d159e383a5557e6c912eb3b + v1.27.1: 23c0b26f9df65671f20c042466c0e6c543e16ba769bbf63aa26abef170f393ba + v1.27.0: 8f99db9aeea00299cb3f28ee61646472014cac91930e4c7551c9153f8f720093 + v1.26.4: cfeca97f1ca612813ae0a56a05d33a9f94e3b1fd8df1debb16f322676819314a + v1.26.3: 942772081d9cd4bd0c07e466439b76a1ca95d3f10a7b53dc524d2946b2b17a71 + v1.26.2: 7e030b2e89d4eb2701d9164e67c804fcb872c29accd76f29bcc148a86a920531 + v1.26.1: cc2fc263f9f88072c744e019ba1c919d9ce2d71603b1b72d288c47c82a86bf08 + v1.26.0: 79837d8b7af95547b92dbab105268dd6382ce2a7afbddad93cc168ab0ca766c8 + ppc64le: + v1.28.1: 0 + v1.28.0: 0 + v1.27.1: 0 + v1.27.0: 0 + v1.26.4: 0 + v1.26.3: 0 + v1.26.2: 0 + v1.26.1: 0 + v1.26.0: 0 +# Checksum +# Kubernetes versions above Kubespray's current target version are untested and should be used with caution. +kubelet_checksums: + arm: + v1.28.2: 0 + v1.28.1: 0 + v1.28.0: 0 + v1.27.6: 0 + v1.27.5: 0 + v1.27.4: 0 + v1.27.3: 0 + v1.27.2: 0 + v1.27.1: 0 + v1.27.0: 0 + v1.26.9: 739c62a6801d935477121614ee3a2ef6deba78ecd088ae5477c3f18bf19d68c8 + v1.26.8: 9db839028b706c005fb6db4442e7dae32c2916acf826a5666d54236399f447fa + v1.26.7: 85fe65155c5bc0dd851d736003ae94e39c03a3e9d65f7435404d177491b8e08d + v1.26.6: 47d9d6e7a26e70b2f446afaa81a219e14cced8046134023637018e5cf36fa0d8 + v1.26.5: dff080c8fe2b8cce04acc2762452259334b233ac41f557588663daae362db5e2 + v1.26.4: cf78ddc97894d518408bc33ec99e2f4e744d7ab26e598fa6a053b09296c80d00 + v1.26.3: 2c862e06293db71a3644728519818a5448db87347ce5862045e5f3eca6ec13e2 + v1.26.2: 24af93f03e514bb66c6bbacb9f00f393ed57df6e43b2846337518ec0b4d1b717 + v1.26.1: fe940be695f73c03275f049cb17f2bf2eb137014334930ce5c6de12573c1f21f + v1.26.0: cabf702fc542fcbb1173c713f1cbec72fd1d9ded36cdcdbbd05d9c308d8360d1 + arm64: + v1.28.2: 32269e9ec38c561d028b65c3048ea6a100e1292cbe9e505565222455c8096577 + v1.28.1: 9b7fa64b2785da4a38768377961e227f8da629c56a5df43ca1b665dd07b56f3c + v1.28.0: 05dd12e35783cab4960e885ec0e7d0e461989b94297e7bea9018ccbd15c4dce9 + v1.27.6: be579ef4e8fa3e1de9d40a77e4d35d99e535a293f66bf3038cbea9cf803d11e5 + v1.27.5: 4e78fafdeb5d61ab6ebcd6e75e968c47001c321bec169bb9bd9f001132de5321 + v1.27.4: c75ad8e7c7ef05c0c021b21a9fe86e92f64db1e4c1bc84e1baf45d8dbb8ba8d1 + v1.27.3: 2838fd55340d59f777d7bd7e5989fc72b7a0ca198cf4f3f723cd9956859ce942 + v1.27.2: 810cd9a611e9f084e57c9ee466e33c324b2228d4249ff38c2588a0cc3224f10d + v1.27.1: dbb09d297d924575654db38ed2fc627e35913c2d4000c34613ac6de4995457d0 + v1.27.0: 37aa2edc7c0c4b3e488518c6a4b44c8aade75a55010534ee2be291220c73d157 + v1.26.9: f6b1dcee9960ffe6b778dc91cabef8ce4a7bd06c76378ef2784232709eace6a5 + v1.26.8: 0f15e484c4a7a7c3bad9e0aa4d4334ca029b97513fbe03f053201dd937cf316e + v1.26.7: 73e086cfd8cd1cef559e739e19aff2932f8a9e0bdba3c9faeb9185a86d067fbb + v1.26.6: 44c2cd64e1317df8252bca1cf196227c543005a3b10d52fb114401cb1617f32f + v1.26.5: 4256e46eb36bea3c31b0372c4d5b669964a2cfb1eabb7e0e2e0dcb1cdd81f2e8 + v1.26.4: a925a5d20d29c362f0c4d60cb005f21d44576837510e0bc65c817961969b4e7e + v1.26.3: d360f919c279a05441b27178030c3d17134c1f257c95f4b22bdb28c2290993e7 + v1.26.2: 33e77f93d141d3b9e207ae50ff050186dea084ac26f9ec88280f85bab9dad310 + v1.26.1: f4b514162b52d19909cf0ddf0b816d8d7751c5f1de60eda90cd84dcccc56c399 + v1.26.0: fb033c1d079cac8babb04a25abecbc6cc1a2afb53f56ef1d73f8dc3b15b3c09e + amd64: + v1.28.2: 17edb866636f14eceaad58c56eab12af7ab3be3c78400aff9680635d927f1185 + v1.28.1: 2bc22332f44f8fcd3fce57879fd873f977949ebd261571fbae31fbb2713a5dd3 + v1.28.0: bfb6b977100963f2879a33e5fbaa59a5276ba829a957a6819c936e9c1465f981 + v1.27.6: daa42f9b6f5e2176bbce0d24d89a05613000630bcddec1fafd2a8d42a523ce9d + v1.27.5: 66df07ab4f9d72028c97ec7e5eea23adc0ab62a209ba2285431456d7d75a5bb3 + v1.27.4: 385f65878dc8b48df0f2bd369535ff273390518b5ac2cc1a1684d65619324704 + v1.27.3: c0e18da6a55830cf4910ecd7261597c66ea3f8f58cf44d4adb6bdcb6e2e6f0bf + v1.27.2: a0d12afcab3b2836de4a427558d067bebdff040e9b306b0512c93d9d2a066579 + v1.27.1: cb2845fff0ce41c400489393da73925d28fbee54cfeb7834cd4d11e622cbd3a7 + v1.27.0: 0b4ed4fcd75d33f5dff3ba17776e6089847fc83064d3f7a3ad59a34e94e60a29 + v1.26.9: baa2b021ab2f90c342518e2b8981a18de7e1e6b33f11c57e3ff23d40364877a8 + v1.26.8: 1c68a65a6a0c2230325e29da0cc3eaaef9bbf688a7a0bb8243b4a7ebfe0e3363 + v1.26.7: 2926ea2cd7fcd644d24a258bdf21e1a8cfd95412b1079914ca46466dae1d74f2 + v1.26.6: da82477404414eb342d6b93533f372aa1c41956a57517453ef3d39ebbfdf8cc2 + v1.26.5: ad5e318ff0e81bc2bef874b2038489722cfcc117bd31726d0193056458c18bff + v1.26.4: 1e29fe7a097066cfbc1c1d2ab37f8b883c8f3fec414bafe8f2c7b960b0fb60fe + v1.26.3: 992d6298bd494b65f54c838419773c4976aca72dfb36271c613537efae7ab7d2 + v1.26.2: e6dd2ee432a093492936ff8505f084b5ed41662f50231f1c11ae08ee8582a3f5 + v1.26.1: 8b99dd73f309ca1ac4005db638e82f949ffcfb877a060089ec0e729503db8198 + v1.26.0: b64949fe696c77565edbe4100a315b6bf8f0e2325daeb762f7e865f16a6e54b5 + ppc64le: + v1.28.2: 79f568ac700d29f88d669c6b6a09adb3b726bdd13c10aa0839cbc70b414372e5 + v1.28.1: 547fc76f0c1d78352fad841ebeacd387fe48750b2648565dfd49197621622fbb + v1.28.0: 22de59965f2d220afa24bf04f4c6d6b65a4bb1cd80756c13381973b1ac3b4578 + v1.27.6: 1001da3586a3f868c371aefde991af94ca780ec1599c8a969390ba105aaf9dcb + v1.27.5: 3c643564bf07753c1388096aef9125811800fd28aa6a5faf3bfb1cef0e1637eb + v1.27.4: 16c69a941f2b67fef35d84062626622d205f9e2375a8daf3410fb1a42fc6e9e7 + v1.27.3: a8ea8f9e857d1140b569fff88d8d750dccaea0aa33d624befbb67f725b5340a0 + v1.27.2: 3af92edd687f7932e7fce877944dfe5efa437bf5f171fc8331725c631a1a86ef + v1.27.1: 7a800b9539beaba0b5d6357070a40fb3c4d216c2ad6693b15f9b1307b1c99e1f + v1.27.0: 17c061a9f7919697ac71c151c19337f65b86f59f59441687ac92e977d851c75b + v1.26.9: e87a83c1ca74e55cea51eda53d29324de7fb7f9330c266ea1f2e270fe0f9b677 + v1.26.8: 92c8deba1f6a89a6d6555c224cebab43d141d5822c252511988ad43ff1a7cc1d + v1.26.7: db5d946bad409a1cea177564fb4111e03e4efc15e86d0078fee401022a4b057a + v1.26.6: 1ca83394b04d3017803a30671eb699a61201e00b656e1fc5b833bd83f8835ff4 + v1.26.5: 787a27855228760a6eeeb200a0e7eab82cb7603b0045ddbadcc1a24f9dc2f178 + v1.26.4: a0d653ed1f5f90d380edb5d6ff77ff61e39e8f1a39dd68719c0126ef6f19c381 + v1.26.3: a12a78b68ec8ac76d482d8a95e0d927ffedac62e630af5fef704f5fecf8e92d9 + v1.26.2: 6f03bc34a34856a3e0e667bea8d6817596c53c7093482d44516163639ce39625 + v1.26.1: bf795bec9b01a9497f46f47b2f3466628fba11f7834c9b2a0f3fa1028c061144 + v1.26.0: df13099611f4eada791e5f41948ef94f7b52c468dff1a6fcad54f4c56b467a7f +kubectl_checksums: + arm: + v1.28.2: 6576aa70413ff00c593a07b549b8b9d9e5ef73c42bb39ab4af475e0fdb540613 + v1.28.1: eaa05dab1bffb8593d8e5caa612530ee5c914ee2be73429b7ce36c3becad893f + v1.28.0: 372c4e7bbe98c7067c4b7820c4a440c931ad77f7cb83d3237b439ca3c14d3d37 + v1.27.6: 3a34a38908a9d0f85dc531cc1c49061ceeaa2ab742382d891d9fc7bf8dc53b8c + v1.27.5: c5e8a02102a93c84413ce8a029f194049429d27ad559061de267d84020a4594b + v1.27.4: 4269133eca9abd29c0a31e15ede2837713635893f1763eccba4b27e66a45abfb + v1.27.3: 1d51a48a0497e47f4f4036687cd337c53d297ea5322e8395432879570379d82e + v1.27.2: 7792f5630543c0af84f444521ee6113da5ae00f2b50872d57324aa725a5341c5 + v1.27.1: fe704e355bf2c5f69964cd12772687535a11a5e9ec0baf4f27e0a8fb156bc615 + v1.27.0: 288470e3eb89a2f55273d753ce6674dfb00e732f2971428acb964810aa726188 + v1.26.9: 8e020ffe72dd4c8694ee5e9f124833ca302a2341fa046650482b38ddb189d1fd + v1.26.8: 411c5c6ba9a247d7fa30f68fd37cfdb92ef14326127bed2512a0daf11a6097d4 + v1.26.7: ad796f714102a78a4f4dfa8f49a3c11cb31a9d74965d6b14f84ef5adb065ed69 + v1.26.6: 1d67bdd384d382decb5b61a3c28d5b05ef0251296551bbc46102c02518c41c02 + v1.26.5: eb3e9bac15ebd31f7e4c21782f60ce9285f59ad4072a78bb53b27029b33609b6 + v1.26.4: 69fccd21b6f7a27d96cda7e6e0cfd3741e3a5bcd7348f2f6e2e9c7550809f030 + v1.26.3: cdb3f670396a775119eb84436d6c0e7e29f24ec511681049200eeb39df9960fb + v1.26.2: a8944021fc9022f73976d8ab2736f21b64b30de3b2a6ccfddd0316ca1d3c6a1d + v1.26.1: e067d59ac19e287026b5c2b75a1077b1312ba82ad64ee01dff2cdafd57720f39 + v1.26.0: 8eef310d0de238c582556d81ab8cbe8d6fca3c0e43ee337a905dcdd3578f9dda + arm64: + v1.28.2: ea6d89b677a8d9df331a82139bb90d9968131530b94eab26cee561531eff4c53 + v1.28.1: 46954a604b784a8b0dc16754cfc3fa26aabca9fd4ffd109cd028bfba99d492f6 + v1.28.0: f5484bd9cac66b183c653abed30226b561f537d15346c605cc81d98095f1717c + v1.27.6: 7322a6f600de6d0d06cf333bdc24cd2a340bba12920b0c2385c97884c808c810 + v1.27.5: 0158955c59c775165937918f910380ed7b52fca4a26fb41a369734e83aa44874 + v1.27.4: 5178cbb51dcfff286c20bc847d64dd35cd5993b81a2e3609581377a520a6425d + v1.27.3: 7bb7fec4e28e0b50b603d64e47629e812408751bd1e0ce059b2fee83b0e3ff6f + v1.27.2: 1b0966692e398efe71fe59f913eaec44ffd4468cc1acd00bf91c29fa8ff8f578 + v1.27.1: fd3cb8f16e6ed8aee9955b76e3027ac423b6d1cc7356867310d128082e2db916 + v1.27.0: f8e09630211f2b7c6a8cc38835e7dea94708d401f5c84b23a37c70c604602ddc + v1.26.9: f945c63220b393ddf8df67d87e67ff74b7f56219a670dee38bc597a078588e90 + v1.26.8: e93f836cba409b5ef5341020d9501067a51bf8210cb35649518e5f4d114244cf + v1.26.7: 71edc4c6838a7332e5f82abb35642ce7f905059a258690b0a585d3ed6de285b3 + v1.26.6: 8261d35cd374c438104bb5257e6c9dafb8443cd0eed8272b219ec5aa17b8ca40 + v1.26.5: c3b3de6a2d7f7e1902c65f6774754e62e86d464ed259509ba29a2a209a515ddf + v1.26.4: eea4054825a4c20cc09bc15abcb1354725ad886338e6892141a071caab91d4b6 + v1.26.3: 0f62cbb6fafa109f235a08348d74499a57bb294c2a2e6ee34be1fa83432fec1d + v1.26.2: 291e85bef77e8440205c873686e9938d7f87c0534e9a491de64e3cc0584295b6 + v1.26.1: 4027cb0a2840bc14ec3f18151b3360dd2d1f6ce730ed5ac28bd846c17e7d73f5 + v1.26.0: 79b14e4ddada9e81d2989f36a89faa9e56f8abe6e0246e7bdc305c93c3731ea4 + amd64: + v1.28.2: c922440b043e5de1afa3c1382f8c663a25f055978cbc6e8423493ec157579ec5 + v1.28.1: e7a7d6f9d06fab38b4128785aa80f65c54f6675a0d2abef655259ddd852274e1 + v1.28.0: 4717660fd1466ec72d59000bb1d9f5cdc91fac31d491043ca62b34398e0799ce + v1.27.6: 2b7adb71c8630904da1b94e262c8c3c477e9609b3c0ed8ae1213a1e156ae38dd + v1.27.5: 9a091fb65e4cf4e8be3ce9a21c79210177dd7ce31a2998ec638c92f37f058bcd + v1.27.4: 4685bfcf732260f72fce58379e812e091557ef1dfc1bc8084226c7891dd6028f + v1.27.3: fba6c062e754a120bc8105cde1344de200452fe014a8759e06e4eec7ed258a09 + v1.27.2: 4f38ee903f35b300d3b005a9c6bfb9a46a57f92e89ae602ef9c129b91dc6c5a5 + v1.27.1: 7fe3a762d926fb068bae32c399880e946e8caf3d903078bea9b169dcd5c17f6d + v1.27.0: 71a78259d70da9c5540c4cf4cff121f443e863376f68f89a759d90cef3f51e87 + v1.26.9: 98ea4a13895e54ba24f57e0d369ff6be0d3906895305d5390197069b1da12ae2 + v1.26.8: d8e0dba258d1096f95bb6746ca359db2ee8abe226e777f89dc8a5d1bb76795aa + v1.26.7: d9dc7741e5f279c28ef32fbbe1daa8ebc36622391c33470efed5eb8426959971 + v1.26.6: ee23a539b5600bba9d6a404c6d4ea02af3abee92ad572f1b003d6f5a30c6f8ab + v1.26.5: 5080bb2e9631fe095139f7e973df9a31eb73e668d1785ffeb524832aed8f87c3 + v1.26.4: 636ac0eaa467dbceda4b2c4e33662adc9709f5ce40341c9fc1a687fc276ac02d + v1.26.3: 026c8412d373064ab0359ed0d1a25c975e9ce803a093d76c8b30c5996ad73e75 + v1.26.2: fcf86d21fb1a49b012bce7845cf00081d2dd7a59f424b28621799deceb5227b3 + v1.26.1: d57be22cfa25f7427cfb538cfc8853d763878f8b36c76ce93830f6f2d67c6e5d + v1.26.0: b6769d8ac6a0ed0f13b307d289dc092ad86180b08f5b5044af152808c04950ae + ppc64le: + v1.28.2: 87cca30846fec99a4fbea122b21e938717b309631bd2220de52049fce30d2e81 + v1.28.1: 81b45c27abbdf2be6c5203dfccfd76ded1ac273f9f7672e6dcdf3440aa191324 + v1.28.0: 7a9dcb4c75b33b9dac497c1a756b1f12c7c63f86fc0f321452360fbe1a79ce0f + v1.27.6: f3ed7752a20dbae271eeff9e9d109381e3ed6772853b5c84dc8a7476bbad847c + v1.27.5: 7ab5fe6eb51bd267b3156ef6e18f9e264e6c7c26ec0dafc2f55edcf3164bac99 + v1.27.4: dee25e38897b16ed9009ddcfd96b6635ab3097a051573c6c444209dc27e8ada5 + v1.27.3: b2da3d262e61ffc3e70511977a933b344b18efa5c238bfa388438bc321bc5e11 + v1.27.2: efee037a276f72c77cc230194d7dadf943a5778be46b7985edeb414d27894266 + v1.27.1: 440bcfd9611319f3d9e5d4fa4cdee2421cdf80c01fad223934d9a9b640673d75 + v1.27.0: daa9f1d4fe3f217de2546bca4ac14601f34b34a25c1f571f1e44eb313aee1385 + v1.26.9: bcb287f24a30bd7ef27bc36dc4f896aba3f1091f947afde73576fbd81af65cc5 + v1.26.8: e94748f8954f44bd5ad5be78a2906ee6a8db7c00ea2d50c9db1bfa09cfc097b9 + v1.26.7: 307eabd20201d1a8f9ac433c03716333565c6bd2532dce4bb42eddc88458d509 + v1.26.6: b56f4422fcf0dc095e777d29eb7eb18cf080098ea47ffdd0a1797a0f8e897fac + v1.26.5: 85f18cad385df01f1758d17c0a0b7f865288121dcc64229a07abb32279b0e44b + v1.26.4: 4f5686ea674d37a639389d95c2a32661986f6f06b530076da2b178839b213414 + v1.26.3: dbfc55dcb86e3e7a2ca01df0317d27b8026861d472bcc7bffa33f45dee693927 + v1.26.2: 88ab57859ac8c4559834d901884ed906c88ea4fa41191e576b34b6c8bd6b7a16 + v1.26.1: 5cfd9fea8dea939a2bd914e1caa7829aa537702ddf14e02a59bf451146942fde + v1.26.0: 9e2b2a03ee5fc726ebd39b6a09498b6508ade8831262859c280d39df33f8830d +kubeadm_checksums: + arm: + v1.28.2: 0 + v1.28.1: 0 + v1.28.0: 0 + v1.27.6: 0 + v1.27.5: 0 + v1.27.4: 0 + v1.27.3: 0 + v1.27.2: 0 + v1.27.1: 0 + v1.27.0: 0 + v1.26.9: a6841e7e554407776e4d0fc83306756ad1836d1f92d6d5cce1055eee1999732a + v1.26.8: 31f37eeed5a9e23719e97055051a5efada2fb69deda958056b3d6b0b41e7eaa5 + v1.26.7: bcff0b0a94f6ee6665a0b1eff0f6aa15ca6caac5040cbbf79cd9bd1125088a5d + v1.26.6: 345775290b5379ab24c6cb333bc26d9bc934a1ce8b795c2948d28608b9d6cb9c + v1.26.5: 7915c12580524fe9e15e35849c77c7a6981849b26ea9324b21efc7b2d66727d1 + v1.26.4: 5517d64c030bc48211b9025b8b70bee1430cc278c3e1dc520967b189f6fa66f9 + v1.26.3: 7ceaf94361e6e7a9f877388df30e424604504de4e36f24aafe30f31f2a27600c + v1.26.2: 84982a359967571baf572aa1a61397906f987d989854ebb67a03b09ea4502af4 + v1.26.1: 0dbd0a197013a3fdc5cb3e012fa8b0d50f38fd3dda56254f4648e08ac867fb60 + v1.26.0: 3368537a5e78fdbfa3cbcae0d19102372a0f4eb6b6a78e7b6b187d5db86d6c9e + arm64: + v1.28.2: 010789a94cf512d918ec4a3ef8ec734dea0061d89a8293059ef9101ca1bf6bff + v1.28.1: 7d2f68917470a5d66bd2a7d62897f59cb4afaeffb2f26c028afa119acd8c3fc8 + v1.28.0: b9b473d2d9136559b19eb465006af77df45c09862cd7ce6673a33aae517ff5ab + v1.27.6: faec35315203913b835e9b789d89001a05e072943c960bcf4de1e331d08e10c8 + v1.27.5: 3023ef1d2eff885af860e13c8b9fcdb857d259728f16bf992d59c2be522cec82 + v1.27.4: b4ede8a18ef3d1cfa61e6fbca8fcab02f8eee3d0770d2329490fa7be90a4cae4 + v1.27.3: 495e2193ed779d25584b4b532796c2270df0f7139ef15fb89dc7980603615ef4 + v1.27.2: 8f01f363f7c7f92de2f2276124a895503cdc5a60ff549440170880f296b087eb + v1.27.1: 024a59cd6fc76784b597c0c1cf300526e856e8c9fefa5fa7948158929b739551 + v1.27.0: acd805c6783b678ee0068b9dd8165bbfd879c345fd9c25d6a978dbc965f48544 + v1.26.9: 14c87cbb9a3fa02308a9546aad192ce2d93e5d1d0296d28ba449079e6a1cb2b2 + v1.26.8: f12d5d748abb8586723b78a2f0300f88faf0391f56d4d49f1ad1cef74160a1b5 + v1.26.7: 34192ceac2287029b36e2d6b682e55dee245ae622701dc3b36bd3203019b18d1 + v1.26.6: 003c7740750ad92d2ff3d58d4a15015906c120c93c7aa605ba98edd936061542 + v1.26.5: d7eede9b44850e16cbe4bb8946a79c03c2c0272f7adc726e63b3a1ac09f13b55 + v1.26.4: a97052d393e60027c354e97c88493aa14a76c8cfb7418bbdf8425b3711d86e3a + v1.26.3: e9a7dbca77f9576a98af1db8747e9dc13e930e40295eaa259dd99fd6e17a173f + v1.26.2: f210d8617acf7c601196294f7ca97e4330b75dad00df6b8dd12393730c501473 + v1.26.1: db101c4bb8e33bd69241de227ed317feee6d44dbd674891e1b9e11c6e8b369bb + v1.26.0: 652844c9518786273e094825b74a1988c871552dc6ccf71366558e67409859d1 + amd64: + v1.28.2: 6a4808230661c69431143db2e200ea2d021c7f1b1085e6353583075471310d00 + v1.28.1: 6134dbc92dcb83c3bae1a8030f7bb391419b5d13ea94badd3a79b7ece75b2736 + v1.28.0: 12ea68bfef0377ccedc1a7c98a05ea76907decbcf1e1ec858a60a7b9b73211bb + v1.27.6: 2bcdd68957ec25d0689bb56f32b4ec86e38463d2691d5ea21cd109c7afa3aa7c + v1.27.5: 35df8efa6e1bc864ed3c48a665caed634a5c46cfd7f41cda5ad66defdfddb2aa + v1.27.4: 7be21d6fb3707fbbe8f0db0403db6234c8af773b941f931bf8248759ee988bcd + v1.27.3: 2cd663f25c2490bd614a6c0ad9089a47ef315caf0dbdf78efd787d5653b1c6e3 + v1.27.2: 95c4bfb7929900506a42de4d92280f06efe6b47e0a32cbc1f5a1ed737592977a + v1.27.1: c7d32d698e99b90f877025104cb4a9f3f8c707e99e6817940f260135b6d1ad0a + v1.27.0: 78d0e04705a7bdb76a514d60f60c073b16334b15f57ee87f064354ca8a233e80 + v1.26.9: 73e128821dd1f799a75c922218d12f6c4618b8e29cc7dae2a7390fb80092d3d9 + v1.26.8: 233a89277ca49dbd666b7391c6c0e43c33d2f08052d5b93e9cd0100ee69430c8 + v1.26.7: 812e6d0e94a3fc77d3e9d09dbe709190b77408936cc4e960d916e8401be11090 + v1.26.6: ba699c3c26aaf64ef46d34621de9f3b62e37656943e09f23dc3bf5aa7b3f5094 + v1.26.5: 793767419c382bae2dc2c9396baafbf051bfa3214accf40dcd7c5ea405583802 + v1.26.4: aa1a137aa2c3427f199ff652c96b11d6b124358296996eb7b8cbde220607b2fe + v1.26.3: 87a1bf6603e252a8fa46be44382ea218cb8e4f066874d149dc589d0f3a405fed + v1.26.2: 277d880dc6d79994fd333e49d42943b7c9183b1c4ffdbf9da59f806acec7fd82 + v1.26.1: 1531abfe96e2e9d8af9219192c65d04df8507a46a081ae1e101478e95d2b63da + v1.26.0: 72631449f26b7203701a1b99f6914f31859583a0e247c3ac0f6aaf59ca80af19 + ppc64le: + v1.28.2: fdc28482a4316c84d61b0997c29c4d4c7b11459af9c654fdee3b4a3031f0fcb7 + v1.28.1: 73e06f2b614ed5665951f7c059e225a7b0b31319c64a3f57e146fbe7a77fe54e + v1.28.0: 146fe9194486e46accd5054fa93939f9608fdbeefefc4bc68e4c40fb4a84ccc9 + v1.27.6: f2b53fdcd0a71390e84d16facbcd7a581f1309cb8bd0501f9508ebefe5a3498c + v1.27.5: 3df86ca5de57a6c6b4043be3c050ed9ed39a50720364b399e12e9e52e87e377b + v1.27.4: 1635ba4269daf422be112ae8c3954332e69c2b1e50ecd285343f1f2d65955de8 + v1.27.3: 3f174f096a5aaa62fe0298e9a16b3af9031cb1d2a29fc3823f80f9a2144d5fd4 + v1.27.2: 412bccd310f4976201d359f0637745944944c0fb2ace315e5e07b180445530c7 + v1.27.1: d4c46dcc3d210b6eae0b8c34b3ece9f24b1bb2697175615c451db717a99430fb + v1.27.0: cf2860aef800496fee0d9fd8722bd7d17c6609e32d87ca380127151f2ce02bb0 + v1.26.9: 1cd0e3623b93aa1786dddb73570a841323db35df4eca45004db2046550ca5d12 + v1.26.8: c93248ce2c9906d16fcb7590d8f3929406b28967da79d6a01c2b2d39203a7f58 + v1.26.7: f8d35a26349c28a01244cfb2f0a163c11daa6bd501e64ce261455c38ffd29bc5 + v1.26.6: d09b5e2221c26f47e2f048b0c375540db14090a4d7b71a708fe51a2b3d0e2b81 + v1.26.5: d6296ca1be9ab7914e9fcd770ce46184db41f5613cec5b1b3de9d51439052fba + v1.26.4: f573ce081e884cc642750f8915d3fdf0ce5696c0d5b4f918d0ff20e76e482739 + v1.26.3: 80b00286e54a87645908c7fd284caef0b1cd7fab5a1948518a6a8d6b0852d49d + v1.26.2: f5d610c4a8a4f99ac6dd07f8cbc0db1de602d5a8895cdaa282c72e36183e310b + v1.26.1: 89ad4d60d266e32147c51e7fb972a9aa6c382391822fa00e27a20f769f3586e8 + v1.26.0: 2431061b3980caa9950a9faaafdfb5cd641e0f787d381db5d10737c03ad800c6 +etcd_binary_checksums: + arm: + v3.5.9: 0 + v3.5.8: 0 + v3.5.7: 0 + v3.5.6: 0 + v3.5.5: 0 + v3.5.4: 0 + v3.5.3: 0 + arm64: + v3.5.9: bb201c106a61bbab59e2d9f37f4bdff99d50201f513c66b4578741eab581fb28 + v3.5.8: 3f4441b293a2d0d4d2f8b2cd9504376e15818f7b865ef4b436e8e6f865f895ff + v3.5.7: 1a35314900da7db006b198dd917e923459b462128101736c63a3cda57ecdbf51 + v3.5.6: 888e25c9c94702ac1254c7655709b44bb3711ebaabd3cb05439f3dd1f2b51a87 + v3.5.5: a8d177ae8ecfd1ef025c35ac8c444041d14e67028c1a7b4eda3a69a8dee5f9c3 + v3.5.4: 8e9c2c28ed6b35f36fd94300541da10e1385f335d677afd8efccdcba026f1fa7 + v3.5.3: 8b00f2f51568303799368ee4a3c9b9ff8a3dd9f8b7772c4f6589e46bc62f7115 + amd64: + v3.5.9: d59017044eb776597eca480432081c5bb26f318ad292967029af1f62b588b042 + v3.5.8: d4c1b8d90ad53658f12ffc293afc5694b7bc6cb093af609188649a799e1cc8dc + v3.5.7: a43119af79c592a874e8f59c4f23832297849d0c479338f9df36e196b86bc396 + v3.5.6: 4db32e3bc06dd0999e2171f76a87c1cffed8369475ec7aa7abee9023635670fb + v3.5.5: 7910a2fdb1863c80b885d06f6729043bff0540f2006bf6af34674df2636cb906 + v3.5.4: b1091166153df1ee0bb29b47fb1943ef0ddf0cd5d07a8fe69827580a08134def + v3.5.3: e13e119ff9b28234561738cd261c2a031eb1c8688079dcf96d8035b3ad19ca58 + ppc64le: + v3.5.9: 551539ebb344ebdc77f170ea51512a6cda35877ffdcbd8b3316b2495a8b2bd87 + v3.5.8: 20e28302c1424b1a3daf7d817f2662e4c64e395a82765d1696cb53cb6bc37a4e + v3.5.7: e861aa6acd4d326ec01bfa06fffb80d33f3f8c26e0eb8b73e4424578d149bd04 + v3.5.6: e235cb885996b8aac133975e0077eaf0a2f8dc7062ad052fa7395668a365906b + v3.5.5: 08422dffd5749f0a5f18bd820241d751e539a666af94251c3715cba8f4702c42 + v3.5.4: 2f0389caed87c2504ffc5a07592ca2a688dee45d599073e5f977d9ce75b5f941 + v3.5.3: f14154897ca5ad4698383b4c197001340fbe467525f6fab3b89ee8116246480f +cni_binary_checksums: + arm: + v1.3.0: 86c4c866a01a8073ad14f6feec74de1fd63669786850c7be47521433f9570902 + v1.2.0: fde5bf2da73995196d248177ee8deeafa8005f33cbe1ab33bd2d75c17ca5a99a + v1.1.1: 84f97baf80f9670a8cd0308dedcc8405d2bbc65166d670b48795e0d1262b4248 + v1.1.0: 91e03a9287dcf8d0249159c90357b0f871ecf7ef0ca5014b2e143f2b30ae9c6d + v1.0.1: d35e3e9fd71687fc7e165f7dc7b1e35654b8012995bbfd937946b0681926d62d + v1.0.0: 910c2ba8b6f50b1081b219d6db04459b555940973249fcf39a792932a91f6d39 + v0.9.1: 909e800d01cc61ffa26f2629e4a202a58d727e6ccaabd0310ef18d2b1e00943c + arm64: + v1.3.0: de7a666fd6ad83a228086bd55756db62ef335a193d1b143d910b69f079e30598 + v1.2.0: 525e2b62ba92a1b6f3dc9612449a84aa61652e680f7ebf4eff579795fe464b57 + v1.1.1: 16484966a46b4692028ba32d16afd994e079dc2cc63fbc2191d7bfaf5e11f3dd + v1.1.0: 33fc7b8d9d5be2d7f95e69e6a9e2af206879942f1e6b7615c04017dce5067f1a + v1.0.1: 2d4528c45bdd0a8875f849a75082bc4eafe95cb61f9bcc10a6db38a031f67226 + v1.0.0: 736335bc5923a37cfb6cc2305489ce6206bcc565004f525b5f7c3604f092aa3a + v0.9.1: ef17764ffd6cdcb16d76401bac1db6acc050c9b088f1be5efa0e094ea3b01df0 + amd64: + v1.3.0: 754a71ed60a4bd08726c3af705a7d55ee3df03122b12e389fdba4bea35d7dd7e + v1.2.0: f3a841324845ca6bf0d4091b4fc7f97e18a623172158b72fc3fdcdb9d42d2d37 + v1.1.1: b275772da4026d2161bf8a8b41ed4786754c8a93ebfb6564006d5da7f23831e5 + v1.1.0: 05d46ac19d01669d424ee57401c0deba101763ac494858064b4ea4ffdcc37c5d + v1.0.1: 5238fbb2767cbf6aae736ad97a7aa29167525dcd405196dfbc064672a730d3cf + v1.0.0: 5894883eebe3e38f4474810d334b00dc5ec59bd01332d1f92ca4eb142a67d2e8 + v0.9.1: 962100bbc4baeaaa5748cdbfce941f756b1531c2eadb290129401498bfac21e7 + ppc64le: + v1.3.0: 8ceff026f4eccf33c261b4153af6911e10784ac169d08c1d86cf6887b9f4e99b + v1.2.0: 4960283b88d53b8c45ff7a938a6b398724005313e0388e0a36bd6d0b2bb5acdc + v1.1.1: 1551259fbfe861d942846bee028d5a85f492393e04bcd6609ac8aaa7a3d71431 + v1.1.0: 98239a57452e93c0a27ba9f87bcbb80c7f982f225246f3fe4f3f5ac9b6b1becb + v1.0.1: f078e33067e6daaef3a3a5010d6440f2464b7973dec3ca0b5d5be22fdcb1fd96 + v1.0.0: 1a055924b1b859c54a97dc14894ecaa9b81d6d949530b9544f0af4173f5a8f2a + v0.9.1: 5bd3c82ef248e5c6cc388f25545aa5a7d318778e5f9bc0a31475361bb27acefe +calicoctl_binary_checksums: + arm: + v3.26.1: 0 + v3.26.0: 0 + v3.25.2: 0 + v3.25.1: 0 + v3.25.0: 0 + v3.24.6: 0 + v3.24.5: 0 + v3.24.4: 0 + v3.24.3: 0 + v3.24.2: 0 + v3.24.1: 0 + v3.24.0: 0 + v3.23.5: 0 + v3.23.4: 0 + v3.23.3: 0 + v3.23.2: 0 + v3.23.1: 0 + v3.23.0: 0 + v3.22.5: 0 + v3.22.4: 0 + v3.22.3: 0 + arm64: + v3.26.1: bba2fbdd6d2998bca144ae12c2675d65c4fbf51c0944d69b1b2f20e08cd14c22 + v3.26.0: b88c4fd34293fa95d4291b7631502f6b9ad38b5f5a3889bb8012f36f001ff170 + v3.25.2: 1cf28599dc1d52ef7c888731f508662a187129ff7bb3294f58319d79c517085c + v3.25.1: 83084be5de90a94bfd7a10da5758acbf200ddd68fa24ee4e7e1dedc8935aa41d + v3.25.0: 6eda153187ab76821903cf6bb69fe11b016529c3344e2dd1a0f7f3cb3069ded0 + v3.24.6: 98eaeb3d75c7ebb41012641e393a442a509f00572981abcc758668ac0806e1e7 + v3.24.5: 2d56b768ed346129b0249261db27d97458cfb35f98bd028a0c817a23180ab2d2 + v3.24.4: 90ffaf6aab30d5e4c7227cf20a68c7254ec9d871f2e7a4a98ba86a855ee61040 + v3.24.3: dfd74167dd55677a54ac73fd1e3f9391d62cf7f4da210b267d437d4a9b7d4561 + v3.24.2: 6fe53f3ba1c7291e2b1cd15ccb72c393297a668cec46f4aa7137499f68fb37e6 + v3.24.1: b7b1a023ddb81ec32f385f4b90f9a3f415d7fce6242e1ae8ebe5c77b2015209c + v3.24.0: db306755fc9c6a746516eec33337bc102b0d546f6b9fc671795b47d1a878f05d + v3.23.5: 0941ad0deeb03d8fda96340948cdbc15d14062086438150cf3ec5ee2767b22c3 + v3.23.4: c54b7d122d9315bbab1a88707b7168a0934a80c4f2a94c9e871bcc8a8cf11c11 + v3.23.3: 741b222f9bb10b7b5e268e5362796061c8862d4f785bb6b9c4f623ea143f4682 + v3.23.2: 232b992e6767c68c8c832cc7027a0d9aacb29901a9b5e8871e25baedbbb9c64c + v3.23.1: 30f7e118c21ecba445b4fbb27f7ac8bc0d1525ab3c776641433e3b1a3388c65b + v3.23.0: 2afa5795c426faae1fdfd966249f8191929e43d2b94bea268fa9c7ab5a36f6b6 + v3.22.5: f0f6ba82d55c7faa5afb361eb76a78c8e2cf38cd06e0287e03821f77af0c7837 + v3.22.4: e84ba529091818282012fd460e7509995156e50854781c031c81e4f6c715a39a + v3.22.3: 3a3e70828c020efd911181102d21cb4390b7b68669898bd40c0c69b64d11bb63 + amd64: + v3.26.1: c8f61c1c8e2504410adaff4a7255c65785fe7805eebfd63340ccd3c472aa42cf + v3.26.0: 19ce069f121f9e245f785a7517521e20fe3294ce1add9d1b2bbcbb0a9b9de24e + v3.25.2: b6f6017b1c9520d8eaea101442d82020123d1efc622964b20d97d3e08e198eed + v3.25.1: 13565e5304209ffaa93df3ba722e6f623b66c76057ca8ff5c5864fa13176fe48 + v3.25.0: 5a464075ccbaa8715882de6b32fe82b41488e904fa66b19c48ee6388cf48b1b8 + v3.24.6: 52e8231d14f626c9b3273659697d95559c72e1b081e713b86eaa7f6910bda384 + v3.24.5: 01e6c8a2371050f9edd0ade9dcde89da054e84d8e96bd4ba8cf82806c8d3e8e7 + v3.24.4: 6d6448537d9abd827c01f289303cf66729578b0bd952c043228568af46000e49 + v3.24.3: 22d7ba5547aff1b4202ddd55952c1e5b6e45e416cd79e1721438aab54a23324a + v3.24.2: 185be69fffcaf46fea8328fcc1b73167021fe16548459148853d084ba8a4aac8 + v3.24.1: 10a36ebc7a4cf355b28e061f5a5f4b261daff4773a51ac73ca1071e7551a934a + v3.24.0: 0da282a6a7870fe25742799a921730343c57a1609c5e255e1bb06b5e85011ee2 + v3.23.5: 4c777881709ddaabcf4b56dcbe683125d7ed5743c036fee9273c5295e522082f + v3.23.4: 1ea0d3b6543645612e8239978878b6adefdb7619a16ecbdb8e6dc2687538f689 + v3.23.3: d9c04ab15bad9d8037192abd2aa4733a01b0b64a461c7b788118a0d6747c1737 + v3.23.2: 3784200cdfc0106c9987df2048d219bb91147f0cc3fa365b36279ac82ea37c7a + v3.23.1: e8fd04d776df5571917512560800bf77f3cdf36ca864c9cae966cb74d62ba4fe + v3.23.0: 38106fdd581ab30dc835efeaf83a88b49b21484f8ad33afbefdaf3c49e007550 + v3.22.5: ba75fa65be0e97555b37282e1ab469ad933866eed164b40513e835279bea7348 + v3.22.4: cc412783992abeba6dc01d7bc67bdb2e3a0cf2f27fc3334bdfc02d326c3c9e15 + v3.22.3: a9e5f6bad4ad8c543f6bdcd21d3665cdd23edc780860d8e52a87881a7b3e203c + ppc64le: + v3.26.1: 7f8baf18f4d7954b6f16b1ddcdadbc818cae2fe1f72137464ccc7b8e6fef03a0 + v3.26.0: b82b931e3aa53248d87b24f00969abfe5ea4518c56a85b5894187c7b47dc452e + v3.25.2: a5e19931ce50953a36387c67b596e56c4b4fc7903893f1ad25248177027ad0dd + v3.25.1: 43f7a19c3f81a658349d727283f201ce5a560dc9a9f7e56d70961755f4196135 + v3.25.0: 15545aa42dfafb12b68070253e649dfbfdb4b495935e4717d2f04c46500d1a9e + v3.24.6: 1d2e2d8ec1524c5fd9f9796bb6ec53e3351d1833c11eb312ca39b549dbccf188 + v3.24.5: 4c40d1703a31eb1d1786287fbf295d614eb9594a4748e505a03a2fbb6eda85b4 + v3.24.4: 2731382823179b49f1e9af7cddca7da191a54d5163a15b19b5ae75ed27dd30f9 + v3.24.3: 2811f71a9a31f8b2965109dc2bbbae24eb5425b4366a6dea8bed1fdb2abe5b60 + v3.24.2: 06c356c1ff741c7d2b49daade10e1bde49ef7db962adfea30dbc4bb314ac8abb + v3.24.1: f2b24bb1cb33795ceba2988ea89e78f25e8fb6283ea22a59a676e37a68a5771e + v3.24.0: cf63f4820e792c101940af3ed6422e1b8769ffcbafd0c3672f2e86675733b053 + v3.23.5: 1b352e73515cbe5746f9b9d7633d8317bd48f713b9b731837f7d79089463321c + v3.23.4: cdd6eace3dc2676b7eed79c665cb0b3dbdd9dcb3bf5b09d7ae20f4f015f75f9b + v3.23.3: f83efcd8d3d7c96dfe8e596dc9739eb5d9616626a6afba29b0af97e5c222575a + v3.23.2: d9ded02381a0fc1311561d0cc9eed9ea827462f3b823593d6ac8bd0591d2020f + v3.23.1: ef5e9b413fbe32da09023cdafc2c3977627dd64a0abcfc68398d3b3923cdd8a6 + v3.23.0: 8b4d40a4613cbc94540b1c7f3252b5924cb549085e73f49e4f84e7814bac7c06 + v3.22.5: 1b3ea734a474d4504c019a8b2213385c8c18cd334edcaefb877e59f8381d2b45 + v3.22.4: f8672ac27ab72c1b05b0f9ae5694881ef8e061bfbcf551f964e7f0a37090a243 + v3.22.3: 7c2fe391f2a18eccff65c64bf93133dc5c58c7322cbd31ea207bbfef5b563947 +ciliumcli_binary_checksums: + arm: + v0.15.0: c8c2d7e2564b1cb6bc82266132f584cb42a430930967ef1fced0b01c8384fedc + v0.14.8: d93dc926c795696f43e7f979ca3a1ed3e912d2c8fd5af305f78c8b08521ef939 + v0.14.7: f9ecd2c029f69d89cb418f461a6098039824ae841aaf9d213df698ccb6a1bfc6 + v0.14.6: 2481dd4edeb01de08c193a421e1b12068b215ee03ff77c3c8ed4514fb810a9d7 + v0.14.5: 08ba6fed412d8e0d1d8f2d9c402aea6c69ac69630f6bf0fd985ad1909b298aae + v0.14.4: 08decd0cca8e1dfcda4322e76c1dbb7eb2c32a2ee6fb8b78d4d6d5bf9cf06373 + v0.14.3: 172e8320ac42750b3a2f41cc6407e4a63c59b30c32ab6e0ff8d5c0695026a5f3 + v0.14.2: 294f27672ad32d065f8899899c68f2561293b68714a65229394769a209254bbd + v0.14.1: 78eae78564624b1346998da9b7da200e5f8489575da09af042bcde0532674264 + v0.14.0: f1873e41a14c380971ee418d60cf37bd8dd2ceefd7eeff6befbbe0768283a65b + arm64: + v0.15.0: 57daf587073147402421f5d8bf069018f73cb66a2da5b4393b742ef59ee15139 + v0.14.8: 25e568d7f26c2a0e83e125a98255ae2faa8ea9e0b0e6d34cd8c690238911e97b + v0.14.7: ad12a491a71185e9cf37665734bf529c5b992025f00c5f3cebfca9524af36472 + v0.14.6: c8218c246fe4c2c2fabfdaaa6f779ab3d8a20bdad9d7113289e9cf9aa0fe75ff + v0.14.5: 7cf13e10162f4ff6114e17c85377b96ed91f187e3131e0c6c35d8f6a181de07d + v0.14.4: 6b6326c178df30085da0f584ac380b1c81bdfbbb7c2a8df0862184444b8bd9eb + v0.14.3: 36df943449dde3eecf1e45cc42a244ef5163ac89f614791a0657cf03ff92273f + v0.14.2: bf94d6aaaf02a6bf728e4ea022e7e37b2dfdc49d5931245be298b2ff4d6af008 + v0.14.1: a73afb03a9815297e6f891aa8420ea04434b479e2777b04b49084d7a8d9cf062 + v0.14.0: e59bd6a38a9bc42f61e34907698ae5cb53a43d93bdec6e7327613f911cb8f205 + amd64: + v0.15.0: 504bbb94b55d4605157b78bf7747cca778888910f8c65729fe69cb94c3d37f5b + v0.14.8: b36014107cea29bdd1df34aa1f292eca9f966d0cc9255232891c0ac6956d421c + v0.14.7: 687b913840f6d54c80e540fac31dd22edbce8962fe8875810e7ed4abc874a45a + v0.14.6: 83fa27d0318f85df78ae6ca06f33c71b900e309cf7488c0db9b9ad5753f6560f + v0.14.5: e6d3b2d297129b10f5690558a85e97d2af407ac30d85758ff77dea686b9c1303 + v0.14.4: 6b3950f8c3b1e8cf7e2123bb4cb1fae4217d720b3353bf924c78d87824c9f1b0 + v0.14.3: 613ffc1cc62ce35b519feb6fc39d1cb2b46635511d365db0da5df498fc6bc001 + v0.14.2: 7ff65f0e85af5daa755c63851f85dea656a59ae4a306e1e9ae02abdf0014f564 + v0.14.1: 86c27fff43f99719271f54a330374ee23f4308aeb6decf7747b354e885a0fcfd + v0.14.0: 73bcbce6fac15c3a62d2a68629f292fa2787440a15998d8c868dae20a6e0e6ed + ppc64le: + v0.15.0: 0 + v0.14.8: 0 + v0.14.7: 0 + v0.14.6: 0 + v0.14.5: 0 + v0.14.4: 0 + v0.14.3: 0 + v0.14.2: 0 + v0.14.1: 0 + v0.14.0: 0 +calico_crds_archive_checksums: + v3.26.1: 6d0afbbd4bdfe4deb18d0ac30adb7165eb08b0114ec5a00d016f37a8caf88849 + v3.26.0: a263e507e79c0a131fdc2a49422d3bdf0456ea5786eb44ace2659aba879b5c7c + v3.25.2: 6a6e95a51a8ebf65d41d671f20854319cca1f26cd87fbcfc30d1382a06ecfee0 + v3.25.1: 4d6b6653499f24f80a85a0a7dac28d9571cabfa25356b08f3b438fd97e322e2d + v3.25.0: 117b4493ad933f24ea6fb82eabfad300da2dd926995bb8c55336595d38c72881 + v3.24.6: 71644374ae7f50bc17cd79544b07e59a3967d1d43b289ae62d750ce9167312e9 + v3.24.5: 10320b45ebcf4335703d692adacc96cdd3a27de62b4599238604bd7b0bedccc3 + v3.24.4: 7f8e54f50388784b5c17ba20ebfb0b65c6a87291771fe8be300646906aa1558d + v3.24.3: dbe3a48d602a3ac9073185a9e12e2452f4520b3e8c01f1af8603ef7af5e44fe9 + v3.24.2: eb0a57e6eb37c4658aa51ecfc078d1dfacb89da23a10b67fed8ea0e4e9c66eea + v3.24.1: 62c30e126c1595adc851f3df0a69926cc6bf97a7d0d38293f23d2232c6411a31 + v3.24.0: 3c6694779b916fa364592a8e19d45f509c67e7dec64fb4cf09c379e170de7720 + v3.23.5: aca591282d9e10a180a2afb05da6ca8db4dd02b886b4788f4962cf5b37ba1bda + v3.23.4: c8b6b033755416756b2b5ef248332b7c5b660618327cb7f83a80fb949fdc601a + v3.23.3: d25f5c9a3adeba63219f3c8425a8475ebfbca485376a78193ec1e4c74e7a6115 + v3.23.2: 37c429650723c5f12ffc20dd390ead1e10d2b8a955a199666d155115a49b4dcc + v3.23.1: a1754ae4bb158e3b46ba3fb326d8038d54cd0dc2c5c8527eadf2b0a6cf8ef2e3 + v3.23.0: 27dd12ff792eb8f680506566e8d99467673f859298fe93d4f23c2139cc3f0c96 + v3.22.5: f3b6a6861b7beae549b4cf0be5c4b954c0cc19e95adb89dd9d78e983f9f2a5d7 + v3.22.4: e72e7b8b26256950c1ce0042ac85fa83700154dae9723c8d007de88343f6a7e5 + v3.22.3: 55ece01da00f82c62619b82b6bfd6442a021acc6fd915a753735e6ebceabaa21 +krew_archive_checksums: + darwin: + arm: + v0.4.4: 0 + v0.4.3: 0 + v0.4.2: 0 + arm64: + v0.4.4: e6ac776140b228d6bdfda11247baf4e9b11068f42005d0975fc260c629954464 + v0.4.3: 22f29ce3c3c9c030e2eaf3939d2b00f0187dfdbbfaee37fba8ffaadc46e51372 + v0.4.2: a69d48f8cad7d87b379071129cde3ee4abcaaa1c3f3692bc80887178b2cc7d33 + amd64: + v0.4.4: 5f4d2f34868a87cf1188212cf7cb598e76a32f389054089aad1fa46e6daf1e1b + v0.4.3: 6f6a774f03ad4190a709d7d4dcbb4af956ca0eb308cb0d0a44abc90777b0b21a + v0.4.2: 47c6b5b647c5de679a2302444f75a36a70530fa4751cb655e0edd5da56a5f110 + ppc64le: + v0.4.4: 0 + v0.4.3: 0 + v0.4.2: 0 + linux: + arm: + v0.4.4: 4f3d550227e014f3ba7c72031108ffda0654cb755f70eb96be413a5102d23333 + v0.4.3: 68eb9e9f5bba29c7c19fb52bfc43a31300f92282a4e81f0c51ad26ed2c73eb03 + v0.4.2: 115f503e35ef7f63f00a9b01236d80a9f94862ec684010a81c3a3b51bdca1351 + arm64: + v0.4.4: f8f0cdbf698ed3e8cb46e7bd213754701341a10e11ccb69c90d4863e0cf5a16a + v0.4.3: 0994923848882ad0d4825d5af1dc227687a10a02688f785709b03549dd34d71d + v0.4.2: 7581be80d803536acc63cceb20065023b96f07fd7eb9f4ee495dce0294a866eb + amd64: + v0.4.4: e471396b0ed4f2be092b4854cc030dfcbb12b86197972e7bef0cb89ad9c72477 + v0.4.3: 5df32eaa0e888a2566439c4ccb2ef3a3e6e89522f2f2126030171e2585585e4f + v0.4.2: 203bfd8006b304c1e58d9e96f9afdc5f4a055e0fbd7ee397fac9f36bf202e721 + ppc64le: + v0.4.4: 0 + v0.4.3: 0 + v0.4.2: 0 + windows: + arm: + v0.4.4: 0 + v0.4.3: 0 + v0.4.2: 0 + arm64: + v0.4.4: 0 + v0.4.3: 0 + v0.4.2: 0 + amd64: + v0.4.4: da0dfeb2a598f11fb9ce871ee7f3b1a69beb371a45f531ee65a71b2201511d28 + v0.4.3: d1343a366a867e9de60b23cc3d8ee935ee185af25ff8f717a5e696ba3cae7c85 + v0.4.2: 3150ff0291ac876ebe4fe0e813ee90a18aa2bc0510c3adcfae6117dec44ef269 + ppc64le: + v0.4.4: 0 + v0.4.3: 0 + v0.4.2: 0 +helm_archive_checksums: + arm: + v3.12.3: 6b67cf5fc441c1fcb4a860629b2ec613d0e6c8ac536600445f52a033671e985e + v3.12.2: 39cc63757901eaea5f0c30b464d3253a5d034ffefcb9b9d3c9e284887b9bb381 + v3.12.1: 6ae6d1cb3b9f7faf68d5cd327eaa53c432f01e8fd67edba4e4c744dcbd8a0883 + v3.12.0: 1d1d3b0b6397825c3f91ec5f5e66eb415a4199ccfaf063ca399d64854897f3f0 + v3.11.3: 0816db0efd033c78c3cc1c37506967947b01965b9c0739fe13ec2b1eea08f601 + v3.11.2: 444b65100e224beee0a3a3a54cb19dad37388fa9217ab2782ba63551c4a2e128 + v3.11.1: 77b797134ea9a121f2ede9d159a43a8b3895a9ff92cc24b71b77fb726d9eba6d + v3.11.0: cddbef72886c82a123038883f32b04e739cc4bd7b9e5f869740d51e50a38be01 + v3.10.3: dca718eb68c72c51fc7157c4c2ebc8ce7ac79b95fc9355c5427ded99e913ec4c + arm64: + v3.12.3: 79ef06935fb47e432c0c91bdefd140e5b543ec46376007ca14a52e5ed3023088 + v3.12.2: cfafbae85c31afde88c69f0e5053610c8c455826081c1b2d665d9b44c31b3759 + v3.12.1: 50548d4fedef9d8d01d1ed5a2dd5c849271d1017127417dc4c7ef6777ae68f7e + v3.12.0: 658839fed8f9be2169f5df68e55cb2f0aa731a50df454caf183186766800bbd0 + v3.11.3: 9f58e707dcbe9a3b7885c4e24ef57edfb9794490d72705b33a93fa1f3572cce4 + v3.11.2: 0a60baac83c3106017666864e664f52a4e16fbd578ac009f9a85456a9241c5db + v3.11.1: 919173e8fb7a3b54d76af9feb92e49e86d5a80c5185020bae8c393fa0f0de1e8 + v3.11.0: 57d36ff801ce8c0201ce9917c5a2d3b4da33e5d4ea154320962c7d6fb13e1f2c + v3.10.3: 260cda5ff2ed5d01dd0fd6e7e09bc80126e00d8bdc55f3269d05129e32f6f99d + amd64: + v3.12.3: 1b2313cd198d45eab00cc37c38f6b1ca0a948ba279c29e322bdf426d406129b5 + v3.12.2: 2b6efaa009891d3703869f4be80ab86faa33fa83d9d5ff2f6492a8aebe97b219 + v3.12.1: 1a7074f58ef7190f74ce6db5db0b70e355a655e2013c4d5db2317e63fa9e3dea + v3.12.0: da36e117d6dbc57c8ec5bab2283222fbd108db86c83389eebe045ad1ef3e2c3b + v3.11.3: ca2d5d40d4cdfb9a3a6205dd803b5bc8def00bd2f13e5526c127e9b667974a89 + v3.11.2: 781d826daec584f9d50a01f0f7dadfd25a3312217a14aa2fbb85107b014ac8ca + v3.11.1: 0b1be96b66fab4770526f136f5f1a385a47c41923d33aab0dcb500e0f6c1bf7c + v3.11.0: 6c3440d829a56071a4386dd3ce6254eab113bc9b1fe924a6ee99f7ff869b9e0b + v3.10.3: 950439759ece902157cf915b209b8d694e6f675eaab5099fb7894f30eeaee9a2 + ppc64le: + v3.12.3: 8f2182ae53dd129a176ee15a09754fa942e9e7e9adab41fd60a39833686fe5e6 + v3.12.2: fb0313bfd6ec5a08d8755efb7e603f76633726160040434fd885e74b6c10e387 + v3.12.1: 32b25dba14549a4097bf3dd62221cf6df06279ded391f7479144e3a215982aaf + v3.12.0: 252d952b0e1b4ed2013710ddedf687ed5545d9f95a4fd72de0ff9617ff69155c + v3.11.3: 9f0a8299152ec714cee7bdf61066ba83d34d614c63e97843d30815b55c942612 + v3.11.2: 04cbb8d053f2d8023e5cc6b771e9fa384fdd341eb7193a0fb592b7e2a036bf3d + v3.11.1: 6ab8f2e253c115b17eda1e10e96d1637047efd315e9807bcb1d0d0bcad278ab7 + v3.11.0: 6481a51095f408773212ab53edc2ead8a70e39eba67c2491e11c4229a251f9b5 + v3.10.3: 93cdf398abc68e388d1b46d49d8e1197544930ecd3e81cc58d0a87a4579d60ed +cri_dockerd_archive_checksums: + arm: + 0.3.4: 0 + 0.3.3: 0 + 0.3.2: 0 + 0.3.1: 0 + 0.3.0: 0 + 0.2.6: 0 + 0.2.5: 0 + arm64: + 0.3.4: 598709c96585936729140d31a76be778e86f9e31180ff3622a44b63806f37779 + 0.3.3: fa0aa587fc7615248f814930c2e0c9a252afb18dc37c8f4d6d0263faed45d5a7 + 0.3.2: b24ae82808bb5ee531348c952152746241ab9b1b7477466ba6c47a7698ef16ae + 0.3.1: dcaa2794ac23348c6d370717a68e70d1da1723a11a892d63459cd88fb5d82226 + 0.3.0: 2a7e5bb156b80f737ef07ae2e8050394ea3e47fb0b7055afac47a365eaa321fb + 0.2.6: 90122641e45e8ff81dbdd4d84c06fd9744b807b87bff5d0db7f826ded326a9fd + 0.2.5: 067242bf5e4b39fece10500a239612c7b0723ce9766ba309dbd22acaf1a2def2 + amd64: + 0.3.4: b77a1fbd70d12e5b1dacfa24e5824619ec54184dbc655e721b8523572651adeb + 0.3.3: 169dce95e7252165c719e066a90b4a64af64119f9ee74fdca73bf9386bcf96c8 + 0.3.2: 93acc0b8c73c68720c9e40b89c2a220a2df315eb2cd3d162b294337c4dcb2193 + 0.3.1: 126431e7b207e013004311f5a21803cad44511616e7440157381476bdc6c5219 + 0.3.0: 8e6a445591e77b9570299d0afadeee26cb7aa23e4bfd7518baa6a3260b9ee889 + 0.2.6: 5d57b160d5a1f75333149823bec3e291a1a0960383ddc9ddd6e4ff177382c755 + 0.2.5: 1660052586390fd2668421d16265dfcc2bbdba79d923c7ede268cf91935657c1 + ppc64le: + 0.3.4: 0 + 0.3.3: 0 + 0.3.2: 0 + 0.3.1: 0 + 0.3.0: 0 + 0.2.6: 0 + 0.2.5: 0 +runc_checksums: + arm: + v1.1.9: 0 + v1.1.8: 0 + v1.1.7: 0 + v1.1.6: 0 + v1.1.5: 0 + v1.1.4: 0 + v1.1.3: 0 + arm64: + v1.1.9: b43e9f561e85906f469eef5a7b7992fc586f750f44a0e011da4467e7008c33a0 + v1.1.8: 7c22cb618116d1d5216d79e076349f93a672253d564b19928a099c20e4acd658 + v1.1.7: 1b309c4d5aa4cc7b888b2f79c385ecee26ca3d55dae0852e7c4a692196d5faab + v1.1.6: da5b2ed26a173a69ea66eae7c369feebf59c1031e14985f512a0a293bb5f76fb + v1.1.5: 54e79e4d48b9e191767e4abc08be1a8476a1c757e9a9f8c45c6ded001226867f + v1.1.4: dbb71e737eaef454a406ce21fd021bd8f1b35afb7635016745992bbd7c17a223 + v1.1.3: 00c9ad161a77a01d9dcbd25b1d76fa9822e57d8e4abf26ba8907c98f6bcfcd0f + amd64: + v1.1.9: b9bfdd4cb27cddbb6172a442df165a80bfc0538a676fbca1a6a6c8f4c6933b43 + v1.1.8: 1d05ed79854efc707841dfc7afbf3b86546fc1d0b3a204435ca921c14af8385b + v1.1.7: c3aadb419e5872af49504b6de894055251d2e685fddddb981a79703e7f895cbd + v1.1.6: 868bee5b8dc2a01df0ca41d0accfad6a3372dc1165ebfb76143d2c6672e86115 + v1.1.5: f00b144e86f8c1db347a2e8f22caade07d55382c5f76dd5c0a5b1ab64eaec8bb + v1.1.4: db772be63147a4e747b4fe286c7c16a2edc4a8458bd3092ea46aaee77750e8ce + v1.1.3: 6e8b24be90fffce6b025d254846da9d2ca6d65125f9139b6354bab0272253d01 + ppc64le: + v1.1.9: 065cf4f84b5acc0acdb017af2955743dfb5f5e1f49a493eea3e8206f33bf6fe6 + v1.1.8: a816cd654e804249c4f757cc6bf2aa2c128e4b8e6a993067d44c63c891c081ab + v1.1.7: eb0e76876d09fa8119dc6e6b037107e5d265d1cfa51f1fbed5418e5745ecf153 + v1.1.6: f98d585dd88d45a296a3f3adde39eaec84e0cfc75f75c50e5470d871e3538460 + v1.1.5: 4f06d25b46e11e6670bf38e638c9183bb6676787801f1226f0aa8e74e40169ea + v1.1.4: 0f7fb3d2426b6012d9b33c354c778c0ffbce02c329c4c16c1189433a958fd60d + v1.1.3: 3b1b7f953fc8402dec53dcf2de05b6b72d86850737efa9766f8ffefc7cae3c0a +crun_checksums: + arm: + 1.8.6: 0 + 1.8.5: 0 + 1.8.4: 0 + 1.8.3: 0 + 1.8.2: 0 + 1.8.1: 0 + 1.7.2: 0 + 1.7.1: 0 + arm64: + 1.8.6: 1f86d20292284f29593594df8d8556d5363a9e087e169626604cc212c77d1727 + 1.8.5: 77032341af7c201db03a53e46707ba8b1af11cdd788530426f2da6ccb9535202 + 1.8.4: 29bbb848881868c58908933bab252e73ee055672d00b7f40cea751441ca74fa4 + 1.8.3: 5394336630618c724274bf3e5e0c8a64c2e67e4723f671029c4f57f459359f73 + 1.8.2: d17970486fab69058e182c3322b7f9fe51561cc3ce28339a0d65b0c81acda933 + 1.8.1: c8382b91a52ac09797ff44990daf014803dde9487d1a41243bc9d8eaf07484e4 + 1.7.2: 576a39ca227a911e0e758db8381d2786f782bfbd40b54684be4af5e1fe67b018 + 1.7.1: 8d458c975f6bf754e86ebedda9927abc3942cbebe4c4cb34a2f1df5acd399690 + amd64: + 1.8.6: 23cd9901106ad7a8ebf33725a16b99a14b95368a085d6ffc2ede0b0c9b002bde + 1.8.5: 75062fa96a7cabd70e6f6baf1e11da00131584cc74a2ef682a172769178d8731 + 1.8.4: 99be7d3c9ba3196c35d64b63fa14e9f5c37d1e91b194cfdbfa92dbcbebd651bc + 1.8.3: f82ccdc575a72fe2d91ea8d68161746a0e28898bc86a2a6f55eed00aa1d79afa + 1.8.2: 9febf1dd7600d15db2ee9a6b8836a76db563bf715e009d0c5f662353e7fa6c29 + 1.8.1: b7f2150da473ed2d052df371244176aa96c9ad908fed06b81ebcb51a8a0f6b06 + 1.7.2: 2bd2640d43bc78be598e0e09dd5bb11631973fc79829c1b738b9a1d73fdc7997 + 1.7.1: 8e095f258eee554bb94b42af07aa5c54e0672a403d56b2cfecd49153a11d6760 + ppc64le: + 1.8.6: 0 + 1.8.5: 0 + 1.8.4: 0 + 1.8.3: 0 + 1.8.2: 0 + 1.8.1: 0 + 1.7.2: 0 + 1.7.1: 0 +youki_checksums: + arm: + 0.1.0: 0 + 0.0.5: 0 + 0.0.4: 0 + 0.0.3: 0 + 0.0.2: 0 + 0.0.1: 0 + arm64: + 0.1.0: 0 + 0.0.5: 0 + 0.0.4: 0 + 0.0.3: 0 + 0.0.2: 0 + 0.0.1: 0 + amd64: + 0.1.0: f00677e9674215b44f140f0c0f4b79b0001c72c073d2c5bb514b7a9dcb13bdbc + 0.0.5: 8504f4c35a24b96782b9e0feb7813aba4e7262c55a39b8368e94c80c9a4ec564 + 0.0.4: c213376393cb16462ef56586e68fef9ec5b5dd80787e7152f911d7cfd72d952e + 0.0.3: 15df10c78f6a35e45a1dce92c827d91b9aef22dc926c619ff5befafc8543f1bb + 0.0.2: dd61f1c3af204ec8a29a52792897ca0d0f21dca0b0ec44a16d84511a19e4a569 + 0.0.1: 8bd712fe95c8a81194bfbc54c70516350f95153d67044579af95788fbafd943b + ppc64le: + 0.1.0: 0 + 0.0.5: 0 + 0.0.4: 0 + 0.0.3: 0 + 0.0.2: 0 + 0.0.1: 0 +kata_containers_binary_checksums: + arm: + 3.1.3: 0 + 3.1.2: 0 + 3.1.1: 0 + 3.1.0: 0 + 3.0.2: 0 + 3.0.1: 0 + 3.0.0: 0 + 2.5.2: 0 + 2.5.1: 0 + 2.5.0: 0 + arm64: + 3.1.3: 0 + 3.1.2: 0 + 3.1.1: 0 + 3.1.0: 0 + 3.0.2: 0 + 3.0.1: 0 + 3.0.0: 0 + 2.5.2: 0 + 2.5.1: 0 + 2.5.0: 0 + amd64: + 3.1.3: 266c906222c85b67867dea3c9bdb58c6da0b656be3a29f9e0bed227c939f3f26 + 3.1.2: 11a2921242cdacf08a72bbce85418fc21c2772615cec6f3de7fd371e04188388 + 3.1.1: 999bab0b362cdf856be6448d1ac4c79fa8d33e79a7dfd1cadaafa544f22ade83 + 3.1.0: 452cc850e021539c14359d016aba18ddba128f59aa9ab637738296d9b5cd78a0 + 3.0.2: a32dc555ffae23f3caab3bc57b03d5ed7792f651221f6cb95cdfe906e18c4bd1 + 3.0.1: e2505482f68cc1b1417b8011f2755bf87171a8dd6daaace28531746118fbddaa + 3.0.0: ff475932f65936504f63ff087c81f89103df2a99e0ceb6571246f63f7a4f948e + 2.5.2: 2c7ce463b32d52b613c1b1ea3d89e83a59ca0fd0ee7fdd24eb854ab2de05ec10 + 2.5.1: 4e4fe5204ae9aea43aa9d9bee467a780d4ae9d52cd716edd7e28393a881377ad + 2.5.0: 044e257c16b8dfa1df92663bd8e4b7f62dbef3e431bc427cdd498ff1b2163515 + ppc64le: + 3.1.3: 0 + 3.1.2: 0 + 3.1.1: 0 + 3.1.0: 0 + 3.0.2: 0 + 3.0.1: 0 + 3.0.0: 0 + 2.5.2: 0 + 2.5.1: 0 + 2.5.0: 0 +gvisor_runsc_binary_checksums: + arm: + 20230807: 0 + 20230801: 0 + 20230731: 0 + 20230724: 0 + 20230717: 0 + 20230710: 0 + 20230627: 0 + 20230621: 0 + 20230605: 0 + 20230529: 0 + 20230522: 0 + 20230517: 0 + 20230508: 0 + 20230501: 0 + 20230417: 0 + arm64: + 20230807: 562c629abb6576d02a4b5a5c32cb4706e29122f72737c55a2bf87d012682117f + 20230801: 69f4b7fd068fcc9a30181657ae5dcdd259e5fe71111d86e7cb0065e190b82fc3 + 20230731: 228ad19507ed23f97d99a2ea19be355f57fa4fddc70d0c425879952bd2d2cd7d + 20230724: bbc929ade0211f1d4759db4d3b1e12942dbd198ec91e3e40a272a787856be6e9 + 20230717: 94e36ee1581b951ab328b097d8aff994e96c035462bd2ea5d67f0bc225996c7a + 20230710: 1af3f8640f517339e2b1d522c6ec7066bc32329d0d96d265af6fe7e6d966d4b3 + 20230627: b791646f8129542f110f5ed9d88c3c1fbe6a242207202a8fba2873fad4c6eca6 + 20230621: 7e57e36c146e4aeae736b777c13bcd077a60110e5f1db9e60b87199aed1b533f + 20230605: ef3a965ff6e585c5604f72172a03e6bf3c7511a04f86925eaad6b78b7b9cb4f1 + 20230529: d31e781026a0afa4e2864839993ab17cf4f581ec92419e7c263f4ed34958a2cd + 20230522: 873163cb0e850685efa2f8c98a3b57502e4af72e5a14edf15c81d8830afa3dc3 + 20230517: 9a107bed8a1184a6f1040e6893c0975e572966a3ecd7009e8b2be70482e4ab1d + 20230508: b1cffc3c3071fe92f2d6c14aa946d50f01b0650ce8a8ed51b240cebc2ae2d1f0 + 20230501: b0e0e74ca92efbb65cfa2de1fbb00f767056c2797ca1b1b091ecee9ae0be8122 + 20230417: 21d01bb86f31812d5bca09fa89129ceee6561e5dd2722afcc52e28649383f311 + amd64: + 20230807: bb5055d820a3698181b593e3f6d2b44e8e957a6df91bea7776fee030c007814f + 20230801: 9df74be6ed44f4b35d5aa5ba1956bb3959680c6909748009a2f9476e04b0921e + 20230731: 50e586ce482bce290d893277644ca950a3ab09b53719170578b324666c6e0eec + 20230724: 859476ba858012724d845b3d417070dfb577b68640f0c0840712485710c975fd + 20230717: 4ab67b11728ab1f8d5a897c3346bfb2d725c73c358a588eab0ad3b3ddaa094c8 + 20230710: cd5648f0e32862a4a733840c565c112fe7b505970a10d3b2e882805158bd82b2 + 20230627: d2db10692a56c73cf2fb78776cb41a911db26f65d122922455786db63e25e22c + 20230621: d4e03fee422c87c8f533493de04286277def528e768b3fc90d22c15d8cc1ff7e + 20230605: 85c8feb5d71f45abe29f97548c5432966cd1e57feab560a923f2ea64395b63c1 + 20230529: 41eb6a5fc545e27c253a21b2d6881c7bfed25932cc34e042023600d17a5b3cd9 + 20230522: 5587bcd68e32432d596492c78e935463a530980c35c34abd215d84e93efd1e8f + 20230517: 20ec2fc2e5bc90840ae6540fcd83879e84c7495b0400bc377f9d502b9fdff591 + 20230508: 2a1385d3ef6e31058671e2d2a1ce83130e934081fa2c5c93589eebf7856f5681 + 20230501: b60dccad63a07553809065d1c4d094dfc5e862353cc61a933b05100ffd1f831c + 20230417: 7c0ccb6144861e45bd14e2ccd02f3fdb935219447256d16c71f6c8e42f90a73d + ppc64le: + 20230807: 0 + 20230801: 0 + 20230731: 0 + 20230724: 0 + 20230717: 0 + 20230710: 0 + 20230627: 0 + 20230621: 0 + 20230605: 0 + 20230529: 0 + 20230522: 0 + 20230517: 0 + 20230508: 0 + 20230501: 0 + 20230417: 0 +gvisor_containerd_shim_binary_checksums: + arm: + 20230807: 0 + 20230801: 0 + 20230731: 0 + 20230724: 0 + 20230717: 0 + 20230710: 0 + 20230627: 0 + 20230621: 0 + 20230605: 0 + 20230529: 0 + 20230522: 0 + 20230517: 0 + 20230508: 0 + 20230501: 0 + 20230417: 0 + arm64: + 20230807: 0b80fba82a7c492dc8c7d8585d172d399116970155c2d35f3a29d37aa5eeb80d + 20230801: 08323b1db170fe611f39306922bf56c8c4ee354d6881385fae0a27d84d6f5a62 + 20230731: 74e72b4c7f0f818f8f1a983072beca3c27f72536be808597fb1f9878f736c6ef + 20230724: 78d106ae19da764065e3e16573f347609206ebc72b48a678e14c2cf5ad04c901 + 20230717: 6425bc2285d8a6322bd9e18864da601b9ba56d7d1759a9a93805e270621036da + 20230710: 3ea3487f5d9ba87fc557c43d8f3b830780696048805883255757c940b669664c + 20230627: 5b0db73bcd6b051e7398fd7bb894bb9969ba7a1291d036aac3d47448e59c5aca + 20230621: bf5fb12a0aba6f3c1520f1b7b18d5bbdc05a5ceeb7143feb46063434c70253de + 20230605: 5ccd58ba24c1b44e6d0add79fd728d124e40b0ccfa40e9731f357719cc020bfc + 20230529: 3b1dd2904e31ab8da3aa518cdee2540caed67eecbe4fc87e143dbe90f294e31d + 20230522: 850bab454c134adefc20bc0f24015beb562ec289537b67c42e6ea3781e1fb763 + 20230517: e3ccb7aea2708d664e2b278b7c40dbf0156e2d2f773e2b13ba5766beb67b7ad2 + 20230508: d16d59d076b0856242d67eda95ee1b2301b04f14abd95ef4fe6c08509f304617 + 20230501: a5f4361897a634ac5832b269e1cc5bc1993825c06e4b0080770a948b36584754 + 20230417: 575163e65e1fda019cb34ee56120d5ecf63b0a3a00dda28c0fc138ce58a5bbff + amd64: + 20230807: fa16b92b3a36665783951fa6abb541bca52983322d135eb1d91ae03b93f4b477 + 20230801: 372a7c60054ab3351f5d32c925fb19bb0467b7eb9cd6032f846ba3f5b23451f8 + 20230731: 8d3f2a3c5634d56a2c2cf6017ccb2ef2764c1bceb3814a1276efe043b440d0b6 + 20230724: ff9c5e51f210c06a3e07a71f6214a0420651cd4c30fddd04d01b6d09ce24eb61 + 20230717: cf342881148b80ed95a8946f9f44781bb9323211e3c7cdc0254d27af5c5b215f + 20230710: db008c53e5a2a868d8b44df0faa051eae7fc5241f1488b53c6040a3665ac04b8 + 20230627: e55b22970b81ccee9b9d0837b11d2a20ad670fc4c028c7e87d03721257e928a9 + 20230621: 180a7320466b997114a903b034fe71aa3a65b9374b3565f7461df587b167e5de + 20230605: 4be04c512c72415b3456b0e94ca9d0c850ffd2c5fd06211f2515da70ac7f74b7 + 20230529: a8fcba918f06d984f4626acef700ab359350d631431564ca37a9ce8b4869eab1 + 20230522: 72587a1f458f20eb029f01a578fe018219688b2c4461453fdb15e97efc987e55 + 20230517: 8de99272600cfdc8b3319dc77f63c57ae592f1319b3c1b50245bc67a0740f2ae + 20230508: 7e4c74b8fc05738a7cdf7c8d99d8030007894cbc7dda774533482c5c459a8ba9 + 20230501: f951c2b8d017005787437d9f849c3edfff54c79d39eb172f92abe745652ef94d + 20230417: 61c3d75a46c8d2652306b5b3ab33e4fb519b03a3852bea738c2700ec650afe4e + ppc64le: + 20230807: 0 + 20230801: 0 + 20230731: 0 + 20230724: 0 + 20230717: 0 + 20230710: 0 + 20230627: 0 + 20230621: 0 + 20230605: 0 + 20230529: 0 + 20230522: 0 + 20230517: 0 + 20230508: 0 + 20230501: 0 + 20230417: 0 +nerdctl_archive_checksums: + arm: + 1.6.0: 20dc5f6912de321d4b6aa8647ce77c261cd6d5726d0b5dfae56bd9cdbd7c28fb + 1.5.0: 36c44498b08a08b652d01812e5f857009373fba64ce9c8ff22e101b205bbc5fb + 1.4.0: b81bece6e8a10762a132f04f54df60d043df4856b5c5ce35d8e6c6936db0b6a0 + 1.3.1: ed24086dbea22612dbcc3d14ee6f1d152b0eb6905bd8c84d3413c1f4c8d45d10 + 1.3.0: 2fa1b0cdb95cafb9ea6293c86a164b2a00b342e02d7a9444794f48f002e187c3 + 1.2.1: 750e4788515779cb8e3ea18678c3125b5724e521214bb18a799903821c17d32c + 1.2.0: d42b3329c3c99d6243a4efa0f33e9324a26597e175dd7fdae574270bf3e26f28 + 1.1.0: cc3bc31b4df015806717149f13b3b329f8fb62e3631aa2abdbae71664ce5c40d + 1.0.0: 8fd283a2f2272b15f3df43cd79642c25f19f62c3c56ad58bb68afb7ed92904c2 + arm64: + 1.6.0: d5f1ed3cda151385d313f9007afc708cae0018c9da581088b092328db154d0c6 + 1.5.0: 1bb613049a91871614d407273e883057040e8393ef7be9508598a92b2efda4b7 + 1.4.0: 0edb064a7d68d0425152ed59472ce7566700b4e547afb300481498d4c7fc6cf1 + 1.3.1: 9e82a6a34c89d3e6a65dc8d77a3723d796d71e0784f54a0c762a2a1940294e3b + 1.3.0: e3405bbaadbee716e50ce4535d03f854129773152aab4876b14f117e1ed3b5ee + 1.2.1: 8dc3d918b44b3ea863a4bc8f121277389d3bdb952d549e44916502a0774ab1bb + 1.2.0: 79d71bbdd0b433838d64ef96d26eb9648911f8d5e5ab494359d32d0ff09abb34 + 1.1.0: a0b57b39341b9d67a3f0ae74e19985c72e930bad14291cbbd8479ed6a6a64e83 + 1.0.0: 27622c9d95efe6d807d5f3770d24ddd71719c6ae18f76b5fc89663a51bcd6208 + amd64: + 1.6.0: fc3e7eef775eff85eb6c16b2761a574e83de444831312bc92e755a1f5577872d + 1.5.0: 6dc945e3dfdc38e77ceafd2ec491af753366a3cf83fefccb1debaed3459829f1 + 1.4.0: d8dcd4e270ae76ab294be3a451a2d8299010e69dce6ae559bc3193535610e4cc + 1.3.1: 3ab552877100b336ebe3167aa57f66f99db763742f2bce9e6233644ef77fb7c9 + 1.3.0: 28299050ed28ed78db4fed95daef1ce326ce0101569dc73cc49f8f7e0c17de25 + 1.2.1: 67aa5cf2a32a3dc0c335b96133daee4d2764d9c1a4d86a38398c4995d2df2176 + 1.2.0: 9d6f3427a1c0af0c38a0a707751b424d04cca13b82c62ad03ec3f4799c2de48c + 1.1.0: fcfd36b0b9441541aab0793c0f586599e6d774781c74f16468a3300026120c0e + 1.0.0: 3e993d714e6b88d1803a58d9ff5a00d121f0544c35efed3a3789e19d6ab36964 + ppc64le: + 1.6.0: c47717ed176f55b291d2068ed6e2445481c391936bd322614e0ff9effe06eb4d + 1.5.0: 169d546d35ba3e6ef088cc81e101f58d5ecb08e71c5ed776c99482854ea3ef8a + 1.4.0: 306d5915b387637407db67ceb96cd89ff7069f0024fb1bbc948a6602638eceaa + 1.3.1: 21700f5fe8786ed7749b61b3dbd49e3f2345461e88fe2014b418a1bdeffbfb99 + 1.3.0: 8eda05b803da56772e38dd0970279e081ba797f6d36ad28a535ba8a98993074c + 1.2.1: d5c60700709e8c3908e9863df57303e43d343e38aadf96f2d06eaaac3bc2b06b + 1.2.0: 64fb56543ee69dafa57691a9dc3b756f9cd6fec021ba1c591680d16ebb8d109d + 1.1.0: 7e97d0a856439d07e82cc26a16dfe243c21da14b7099e330e4da11e825004478 + 1.0.0: 2fb02e629a4be16b194bbfc64819132a72ede1f52596bd8e1ec2beaf7c28c117 +containerd_archive_checksums: + arm: + 1.7.6: 0 + 1.7.5: 0 + 1.7.4: 0 + 1.7.3: 0 + 1.7.2: 0 + 1.7.1: 0 + 1.7.0: 0 + 1.6.24: 0 + 1.6.23: 0 + 1.6.22: 0 + 1.6.21: 0 + 1.6.20: 0 + 1.6.19: 0 + 1.6.18: 0 + 1.6.17: 0 + 1.6.16: 0 + 1.6.15: 0 + 1.6.14: 0 + 1.6.13: 0 + 1.6.12: 0 + 1.6.11: 0 + 1.6.10: 0 + 1.6.9: 0 + 1.6.8: 0 + 1.6.7: 0 + 1.6.6: 0 + 1.6.5: 0 + 1.6.4: 0 + 1.6.3: 0 + 1.6.2: 0 + 1.6.1: 0 + 1.6.0: 0 + 1.5.18: 0 + 1.5.17: 0 + 1.5.16: 0 + 1.5.15: 0 + 1.5.14: 0 + arm64: + 1.7.6: d844a1c8b993e7e9647f73b9814567004dce1287c0529ce55c50519490eafcce + 1.7.5: 98fc6990820d52d45b56ea2cda808157d4e61bb30ded96887634644c03025fa9 + 1.7.4: ea5a04379bd4252fc1e0b7b37f69cd516350c5269054483535d6eab7a0c79d2e + 1.7.3: 85d2eaedabff57ac1d7cd3884bf232155c4c46491f6b071982e4f7b684b74445 + 1.7.2: d75a4ca53d9addd0b2c50172d168b12957e18b2d8b802db2658f2767f15889a6 + 1.7.1: 1f828dc063e3c24b0840b284c5635b5a11b1197d564c97f9e873b220bab2b41b + 1.7.0: e7e5be2d9c92e076f1e2e15c9f0a6e0609ddb75f7616999b843cba92d01e4da2 + 1.6.24: 1d741e9e2d907f02a8b2a46034a28ff9aacdba88c485cef2f4bad18be9ea23ba + 1.6.23: ea7afb82dc5789307e684ef9b4a55ce1ee9a05dc02c2118df640b01207208c45 + 1.6.22: 7882d6e7f4e97dcba041c37592c4cb9e7a5b4d972380c74d959e388b12d57d01 + 1.6.21: d713d8fbec491705ffe8c33ecc9051a904f6eedc92574928e1d33616f291c583 + 1.6.20: c3e6a054b18b20fce06c7c3ed53f0989bb4b255c849bede446ebca955f07a9ce + 1.6.19: 25a0dd6cce4e1058824d6dc277fc01dc45da92539ccb39bb6c8a481c24d2476e + 1.6.18: 56b83a0bc955edc5ebaa3bd0f788e654b63395be00fcb1bd03ff4bdfe4b5e1e7 + 1.6.17: 7e110faa738bff2f5f0ffd54c4ec2c17c05fd2af6de4877c839794ca3dadd61c + 1.6.16: c2bf51fde02ec9cf8b9c18721bc4f53bd1f19fb2bb3251f41ece61af7347e082 + 1.6.15: d63e4d27c51e33cd10f8b5621c559f09ece8a65fec66d80551b36cac9e61a07d + 1.6.14: 3ccb61218e60cbba0e1bbe1e5e2bf809ac1ead8eafbbff36c3195d3edd0e4809 + 1.6.13: 8c7892ae7c2e96a4a9358b1064fb5519a5c0528b715beee67b72e74d7a644064 + 1.6.12: 0a0133336596b2d1dcafe3587eb91ab302afc28f273614e0e02300694b5457a0 + 1.6.11: 1b34d8ff067da482af021dac325dc4e993d7356c0bd9dc8e5a3bb8271c1532de + 1.6.10: 6d655e80a843f480e1c1cead18479185251581ff2d4a2e2e5eb88ad5b5e3d937 + 1.6.9: 140197aee930a8bd8a69ff8e0161e56305751be66e899dccd833c27d139f4f47 + 1.6.8: b114e36ecce78cef9d611416c01b784a420928c82766d6df7dc02b10d9da94cd + 1.6.7: 4167bf688a0ed08b76b3ac264b90aad7d9dd1424ad9c3911e9416b45e37b0be5 + 1.6.6: 807bf333df331d713708ead66919189d7b142a0cc21ec32debbc988f9069d5eb + 1.6.5: 2833e2f0e8f3cb5044566d64121fdd92bbdfe523e9fe912259e936af280da62a + 1.6.4: 0205bd1907154388dc85b1afeeb550cbb44c470ef4a290cb1daf91501c85cae6 + 1.6.3: 354e30d52ff94bd6cd7ceb8259bdf28419296b46cf5585e9492a87fdefcfe8b2 + 1.6.2: a4b24b3c38a67852daa80f03ec2bc94e31a0f4393477cd7dc1c1a7c2d3eb2a95 + 1.6.1: fbeec71f2d37e0e4ceaaac2bdf081295add940a7a5c7a6bcc125e5bbae067791 + 1.6.0: 6eff3e16d44c89e1e8480a9ca078f79bab82af602818455cc162be344f64686a + 1.5.18: 0 + 1.5.17: 0 + 1.5.16: 0 + 1.5.15: 0 + 1.5.14: 0 + amd64: + 1.7.6: 58408cfa025003e671b0af72183b963363d519543d0d0ba186037e9c57489ffe + 1.7.5: 33609ae2d5838bc5798306a1ac30d7f2c6a8cff785ca6253d2be8a8b3ccbab25 + 1.7.4: fc070fabfe3539d46ae5db160d18381270928b3f912e2e800947e9fbd43f510c + 1.7.3: de7f61aacba88ee647a7dcde1ca77672ec44ab9fb3e58ae90c0efc9b2d8f3068 + 1.7.2: 2755c70152ab40856510b4549c2dd530e15f5355eb7bf82868e813c9380e22a7 + 1.7.1: 9504771bcb816d3b27fab37a6cf76928ee5e95a31eb41510a7d10ae726e01e85 + 1.7.0: b068b05d58025dc9f2fc336674cac0e377a478930f29b48e068f97c783a423f0 + 1.6.24: a56fac5ba03c3d6f74ceae14abdc9fafabcba900105e9890c0ac895cc00164ad + 1.6.23: bcf16bb63a295721a2603e9a56602c5d18e5443df04a9f2c1ca5328f41556fcc + 1.6.22: 5671eb4eba97f0ec98223c84401c9aeb21d0ef16ac3ece3eb8fadd46174d7eab + 1.6.21: 04dcc1b99368492caee758583e531392683268197e58156888a3cea2941117b6 + 1.6.20: bb9a9ccd6517e2a54da748a9f60dc9aa9d79d19d4724663f2386812f083968e2 + 1.6.19: 3262454d9b3581f4d4da0948f77dde1be51cfc42347a1548bc9ab6870b055815 + 1.6.18: c4e516376a2392520a87abea94baf2045cc3a67e9e0c90c75fb6ed038170561e + 1.6.17: 5f0584d000769d0cf08fc0e25135614ef5bf52971a6069175c78437699f3b8d4 + 1.6.16: 2415b431a900275c14942f87f751e1e13d513c1c2f062322b5ca5a9a2190f22a + 1.6.15: 191bb4f6e4afc237efc5c85b5866b6fdfed731bde12cceaa6017a9c7f8aeda02 + 1.6.14: 7da626d46c4edcae1eefe6d48dc6521db3e594a402715afcddc6ac9e67e1bfcd + 1.6.13: 97f00411587512e62ec762828e581047b23199f8744754706d09976ec24a2736 + 1.6.12: a56c39795fd0d0ee356b4099a4dfa34689779f61afc858ef84c765c63e983a7d + 1.6.11: 21870d7022c52f5f74336d440deffb208ba747b332a88e6369e2aecb69382e48 + 1.6.10: dd1f4730daf728822aea3ba35a440e14b1dfa8f1db97288a59a8666676a13637 + 1.6.9: 9ee2644bfb95b23123f96b564df2035ec94a46f64060ae12322e09a8ec3c2b53 + 1.6.8: 3a1322c18ee5ff4b9bd5af6b7b30c923a3eab8af1df05554f530ef8e2b24ac5e + 1.6.7: 52e817b712d521b193773529ff33626f47507973040c02474a2db95a37da1c37 + 1.6.6: 0212869675742081d70600a1afc6cea4388435cc52bf5dc21f4efdcb9a92d2ef + 1.6.5: cf02a2da998bfcf61727c65ede6f53e89052a68190563a1799a7298b0cea86b4 + 1.6.4: f23c8ac914d748f85df94d3e82d11ca89ca9fe19a220ce61b99a05b070044de0 + 1.6.3: 306b3c77f0b5e28ed10d527edf3d73f56bf0a1fb296075af4483d8516b6975ed + 1.6.2: 3d94f887de5f284b0d6ee61fa17ba413a7d60b4bb27d756a402b713a53685c6a + 1.6.1: c1df0a12af2be019ca2d6c157f94e8ce7430484ab29948c9805882df40ec458b + 1.6.0: f77725e4f757523bf1472ec3b9e02b09303a5d99529173be0f11a6d39f5676e9 + 1.5.18: d132525d375bbafd3aee8e9aa5517203cef2bbf77197db7522c8730cc526b3db + 1.5.17: b676f56f43bf02782179320b5070fb210cbc1784a9b0875086bc15c3bcc546f3 + 1.5.16: f5326865c2b86aba794f590fd2ca479f817fa1f88c084d97a45816aec9d0ce32 + 1.5.15: 0d09043be08dcf6bf136aa78bfd719e836cf9f9679afa4db0b6e4d478e396528 + 1.5.14: 8513ead11aca164b7e70bcea0429b4e51dad836b6383b806322e128821aaebbd + ppc64le: + 1.7.6: 956fadb01b35c3214f2b6f82abc0dda3e1b754cb223cd24e818334b08cb09fb2 + 1.7.5: 2496e24a95fa74750363a8a7e2ac36acf8d41ee2e4b67a452154ad4c8efbc4bc + 1.7.4: c3397f67fb5756e6336ff76eeb34dfad7ad66235877a4186d82044c7a4caf482 + 1.7.3: d1977922e74147782dd5bb488f260ee14d758d29a7651cd97bc2e6c7cc1a3cce + 1.7.2: cbe7ec913cb603ca218bd8867efdce4bee3b0e0115e467e51c910467daf8184e + 1.7.1: 17d97ef55c6ce7af9778dbafb5e73f577d1b34220043a91cccde49dbcc610342 + 1.7.0: 051e897d3ee5b8c8097f65be447fea2d29226b583ca5d9ed78e9aebcf4e69889 + 1.6.24: abff9e7ec4cc21d19150d2bc55fc89cf53dc03c002cdaf5016ee82aedead9b03 + 1.6.23: 1099fce4a4dfc78712cc1c19be6d9f80e9321f513834dba0b2418bd5c78ad398 + 1.6.22: 0f8647aedd96174704a63a17b1a7cf4c4c5c2fc066606b1a419b2a860b754bfb + 1.6.21: 196d91799070a5ff2f5f3b6efe8516c5377f299d38012b6c0cf4ae77fc8c22c5 + 1.6.20: f3ee666fdd31031f07cbb7bad24f0181fad1094ba36273dc61f8fa5a570b5311 + 1.6.19: 18cf11b6dfc980aca8792a2cd3ea7afed6379c2988ca6fe9e53a19a0bece5a2d + 1.6.18: b7083473061a61200d04f500ad4a96813d1b09f71b2d427019076836c2a49836 + 1.6.17: 2f689ff36ba41c3c86ce926f55b0101118a40dd7b741946386062fddaa287db0 + 1.6.16: 9cfd5dade6a1c2671f5c76496395afe0aa0ce902c13672b306d8d09fdbb99492 + 1.6.15: 502f3e4c8ea2018aaa285fe4f704bfd560fdf93193bb829dd9302d013bc38370 + 1.6.14: 73025da0666079fc3bbd48cf185da320955d323c7dc42d8a4ade0e7926d62bb0 + 1.6.13: f2508ada0c8bd7d3cb09b0e7f10416aba3d643c0da7adc27efe4e76d444322ae + 1.6.12: 088e4d1fe1787fc4a173de24a58da01880d1ead5a13f1ab55e1ade972d3907d4 + 1.6.11: e600a5714ffb29937b3710f9ae81bb7aa15b7b6661192f5e8d0b9b58ac6d5e66 + 1.6.10: 704b1affd306b807fe6b4701d778129283635c576ecedc6d0a9da5370a07d56a + 1.6.9: fe0046437cfe971ef0b3101ee69fcef5cf52e8868de708d35f8b82f998044f6e + 1.6.8: f18769721f614828f6b778030c72dc6969ce2108f2363ddc85f6c7a147df0fb8 + 1.6.7: 0db5cb6d5dd4f3b7369c6945d2ec29a9c10b106643948e3224e53885f56863a9 + 1.6.6: 0 + 1.6.5: 0 + 1.6.4: 0 + 1.6.3: 0 + 1.6.2: 0 + 1.6.1: 0 + 1.6.0: 0 + 1.5.18: 0 + 1.5.17: 0 + 1.5.16: 0 + 1.5.15: 0 + 1.5.14: 0 +skopeo_binary_checksums: + arm: + v1.13.2: 0 + v1.13.1: 0 + v1.13.0: 0 + v1.12.0: 0 + v1.11.2: 0 + v1.11.1: 0 + v1.11.0: 0 + v1.10.0: 0 + v1.9.3: 0 + v1.9.2: 0 + arm64: + v1.13.2: 520cc31c15796405b82d01c78629d5b581eced3512ca0b6b184ed82f5e18dc86 + v1.13.1: 3b7db2b827fea432aa8a861b5caa250271c05da70bd240aa4045f692eba52e24 + v1.13.0: d23e43323c0a441d1825f9da483b07c7f265f2bd0a4728f7daac4239460600a3 + v1.12.0: f34476bd33d2ab3784675611b405cc0855ce5decedfa22287e261d23d17e7688 + v1.11.2: cd90552f7d4eb78ba032c885b47cd97ef015a958279d2a2b828b109d75d6c7e0 + v1.11.1: 693f7d2791e0549b173b2c16f1c3326328aa5e95bc2b4d71f5ecd35b6524b09d + v1.11.0: 3e510999ffb6544b11d339812df75d14a46261518b5c73f530242ebed679fb1b + v1.10.0: 3bfc344d4940df29358f8056de7b8dd488b88a5d777b3106748ba66851fa2c58 + v1.9.3: 27c88183de036ebd4ffa5bc5211329666e3c40ac69c5d938bcdab9b9ec248fd4 + v1.9.2: 1b7b4411c9723dbbdda4ae9dde23a33d8ab093b54c97d3323784b117d3e9413f + amd64: + v1.13.2: 2f00be6ee1c4cbfa7f2452be90a1a2ce88fd92a6d0f6a2e9d901bd2087bd9092 + v1.13.1: 8c15c56a6caffeb863c17d73a6361218c04c7763e020fffc8d5d6745cacfa901 + v1.13.0: 8cb477ee25010497fc9df53a6205dbd9fe264dd8a5ea4e934b9ec24d5bdc126c + v1.12.0: 38143238e945959e6b24dba1447ba49e0b79f10a3aef2634391d0205ab950003 + v1.11.2: c8641decb185f43bb49f3f2a68abbc22e051a497440beba28b10d25d0a856574 + v1.11.1: 9aad99e41533800ce08526a602de2f87b8ce123ea9547358e2cccfa2f9c3a9e0 + v1.11.0: 0a6c0a1b349d2efd2895d2ec9a1d9c5c4bbd59f10c3993acb0c92d31914fbd62 + v1.10.0: 20fbd1bac1d33768c3671e4fe9d90c5233d7e13a40e4935b4b24ebc083390604 + v1.9.3: 6e00cf4661c081fb1d010ce60904dccb880788a52bf10de16a40f32082415a87 + v1.9.2: 5c82f8fc2bcb2502cf7cdf9239f54468d52f5a2a8072893c75408b78173c4ba6 + ppc64le: + v1.13.2: 0 + v1.13.1: 0 + v1.13.0: 0 + v1.12.0: 0 + v1.11.2: 0 + v1.11.1: 0 + v1.11.0: 0 + v1.10.0: 0 + v1.9.3: 0 + v1.9.2: 0 +yq_checksums: + arm: + v4.35.1: b2349bc220394329bc95865375feb5d777f5a5177bcdede272788b218f057a05 + v4.34.2: 161f2b64e7bf277614983014b2b842e9ae9c1f234a9ea12593b0e5ebe5a89681 + v4.34.1: dfda7fc51bdf44d3551c4bca78ecd52c13d7137d99ec3f7b466c50333e0a0b7c + v4.33.3: 77c239e17cb50a330da3c48af7dbd3b667af02c950a9ecae49b171cc0bf66c48 + v4.33.2: ee8d61975ebdfabf9b79ed2947ebfe39d62f290ab71d6eb53af183f06cfa1af2 + v4.33.1: 80ec1487b9497e0c5f182c8e44d73b5e9c437719230e114b9fce2fa25c23f95a + v4.32.2: 776b8dcfaf796255e60c0e9893d24ce9eb61cc7e787a1cc8b4a5dc37a84f02d5 + v4.32.1: 0fc6c3e41af7613dcd0e2af5c3c448e7d0a46eab8b38a22f682d71e35720daed + v4.31.2: 4873c86de5571487cb2dcfd68138fc9e0aa9a1382db958527fa8bc02349b5b26 + arm64: + v4.35.1: 1d830254fe5cc2fb046479e6c781032976f5cf88f9d01a6385898c29182f9bed + v4.34.2: 6ea70418755aa805b6d03e78a1c8a1bf236220f187ba3fb4f30663a35c43b4c1 + v4.34.1: c1410df7b1266d34a89a91dcfeaf8eb27cb1c3f69822d72040d167ec61917ba0 + v4.33.3: 15925a972d268bcb0a7aa2236c7e5925b7a3ba4b5569bb57e943db7e8c6f834f + v4.33.2: aa86e5f36850f9350a7393c7cf654ee726df99ae985b663eb3605ca2bdd24692 + v4.33.1: e3a47e60765322995f11422108829881d2166dcf9b13a3ae8ad2c002ee61f8a1 + v4.32.2: 2f855a9e616eb9c635269630666e7594931ab7524326ff02a8351d5762a28940 + v4.32.1: db4eba6ced2656e1c40e4d0f406ee189773bdda1054cbd097c1dba471e04dd4d + v4.31.2: 590f19f0a696103376e383c719fe0df28c62515627bf44e5e69403073ba83cbf + amd64: + v4.35.1: bd695a6513f1196aeda17b174a15e9c351843fb1cef5f9be0af170f2dd744f08 + v4.34.2: 1952f93323e871700325a70610d2b33bafae5fe68e6eb4aec0621214f39a4c1e + v4.34.1: c5a92a572b3bd0024c7b1fe8072be3251156874c05f017c23f9db7b3254ae71a + v4.33.3: 4ee662847c588c3ef2fec8bfb304e8739e3dbaba87ccb9a608d691c88f5b64dc + v4.33.2: fbcc9551afd66622ffd68ad417837139741b2ad0eef9af1bb4b64e3596679ffa + v4.33.1: 5b9d60aa55e53fc06c9114aa5b9d5f1de9bdb231c91aed62b35d10d991831cda + v4.32.2: 0e5c6b5a74d4ccd6eed43180f60dd48a6e1d0e778f834dca33a312301674b628 + v4.32.1: e53b82caa86477bd96cf447138c72c9a0a857142a5bcdd34440b2644693ed18f + v4.31.2: 71ef4141dbd9aec3f7fb45963b92460568d044245c945a7390831a5a470623f7 + ppc64le: + v4.35.1: 713e2c40c5d659cbed7bf093f4c718674a75f9fe5b10ac96fd422372af198684 + v4.34.2: e149b36f93a1318414c0af971755a1488df4844356b6e9e052adf099a72e3a3a + v4.34.1: 3e629c51a07302920110893796f54f056a6ef232f791b9c67fdbe95362921a03 + v4.33.3: b5b7a59e72a5f603b819f50f2dc3e42e53127398c6f0d77da7a06d2f4d0952ea + v4.33.2: 682db3754da9f91ea447e9811e582c3f395e8d566b5afc43d1a67f6be56fa1e2 + v4.33.1: 22311b8726022963e50494d5510ffe5bb63bd0aa3f47feff445e35c4483ac68a + v4.32.2: a3650f6838a8a7a9b8914a8c27c16fb3a172ac3430836565c3927bf1a641d714 + v4.32.1: 43f1f5078a2fa7748cb5dab693538a9e634557ef2c8aad390f147beb727278cf + v4.31.2: 14e79e8eb6d36858adb3355d77ccd1d128ce74257d1358f53e1a46b2f252e28c diff --git a/kubespray/roles/download/defaults/main/main.yml b/kubespray/roles/download/defaults/main/main.yml new file mode 100644 index 0000000..3c33694 --- /dev/null +++ b/kubespray/roles/download/defaults/main/main.yml @@ -0,0 +1,1173 @@ +--- +local_release_dir: /tmp/releases +download_cache_dir: /tmp/kubespray_cache + +# If this is true, debug information will be displayed but +# may contain some private data, so it is recommended to set it to false +# in the production environment. +unsafe_show_logs: false + +# do not delete remote cache files after using them +# NOTE: Setting this parameter to TRUE is only really useful when developing kubespray +download_keep_remote_cache: false + +# Only useful when download_run_once is false: Localy cached files and images are +# uploaded to kubernetes nodes. Also, images downloaded on those nodes are copied +# back to the ansible runner's cache, if they are not yet preset. +download_force_cache: false + +# Used to only evaluate vars from download role +skip_downloads: false + +# Optionally skip kubeadm images download +skip_kubeadm_images: false +kubeadm_images: {} + +# if this is set to true will only download files once. Doesn't work +# on Flatcar Container Linux by Kinvolk unless the download_localhost is true and localhost +# is running another OS type. Default compress level is 1 (fastest). +download_run_once: false +download_compress: 1 + +# if this is set to true will download container +download_container: true + +# if this is set to true, uses the localhost for download_run_once mode +# (requires docker and sudo to access docker). You may want this option for +# local caching of docker images or for Flatcar Container Linux by Kinvolk cluster nodes. +# Otherwise, uses the first node in the kube_control_plane group to store images +# in the download_run_once mode. +download_localhost: false + +# Always pull images if set to True. Otherwise check by the repo's tag/digest. +download_always_pull: false + +# Some problems may occur when downloading files over https proxy due to ansible bug +# https://github.com/ansible/ansible/issues/32750. Set this variable to False to disable +# SSL validation of get_url module. Note that kubespray will still be performing checksum validation. +download_validate_certs: true + +# Use the first kube_control_plane if download_localhost is not set +download_delegate: "{% if download_localhost %}localhost{% else %}{{ groups['kube_control_plane'][0] }}{% endif %}" + +# Allow control the times of download retries for files and containers +download_retries: 4 + +# The docker_image_info_command might seems weird but we are using raw/endraw and `{{ `{{` }}` to manage the double jinja2 processing +docker_image_pull_command: "{{ docker_bin_dir }}/docker pull" +docker_image_info_command: "{{ docker_bin_dir }}/docker images -q | xargs -i {{ '{{' }} docker_bin_dir }}/docker inspect -f {% raw %}'{{ '{{' }} if .RepoTags }}{{ '{{' }} join .RepoTags \",\" }}{{ '{{' }} end }}{{ '{{' }} if .RepoDigests }},{{ '{{' }} join .RepoDigests \",\" }}{{ '{{' }} end }}' {% endraw %} {} | tr '\n' ','" +nerdctl_image_info_command: "{{ bin_dir }}/nerdctl -n k8s.io images --format '{% raw %}{{ .Repository }}:{{ .Tag }}{% endraw %}' 2>/dev/null | grep -v ^:$ | tr '\n' ','" +nerdctl_image_pull_command: "{{ bin_dir }}/nerdctl -n k8s.io pull --quiet" +crictl_image_info_command: "{{ bin_dir }}/crictl images --verbose | awk -F ': ' '/RepoTags|RepoDigests/ {print $2}' | tr '\n' ','" +crictl_image_pull_command: "{{ bin_dir }}/crictl pull" + +image_command_tool: "{%- if container_manager == 'containerd' -%}nerdctl{%- elif container_manager == 'crio' -%}crictl{%- else -%}{{ container_manager }}{%- endif -%}" +image_command_tool_on_localhost: "{{ image_command_tool }}" + +image_pull_command: "{{ lookup('vars', image_command_tool + '_image_pull_command') }}" +image_info_command: "{{ lookup('vars', image_command_tool + '_image_info_command') }}" +image_pull_command_on_localhost: "{{ lookup('vars', image_command_tool_on_localhost + '_image_pull_command') }}" +image_info_command_on_localhost: "{{ lookup('vars', image_command_tool_on_localhost + '_image_info_command') }}" + +# Arch of Docker images and needed packages +image_arch: "{{ host_architecture | default('amd64') }}" + +# Versions +kubeadm_version: "{{ kube_version }}" +crun_version: 1.8.5 +runc_version: v1.1.9 +kata_containers_version: 3.1.3 +youki_version: 0.1.0 +gvisor_version: 20230807 +containerd_version: 1.7.6 +cri_dockerd_version: 0.3.4 + +# this is relevant when container_manager == 'docker' +docker_containerd_version: 1.6.16 + +# gcr and kubernetes image repo define +gcr_image_repo: "gcr.io" +kube_image_repo: "registry.k8s.io" + +# docker image repo define +docker_image_repo: "docker.io" + +# quay image repo define +quay_image_repo: "quay.io" + +# github image repo define (ex multus only use that) +github_image_repo: "ghcr.io" + +# TODO(mattymo): Move calico versions to roles/network_plugins/calico/defaults +# after migration to container download +calico_version: "v3.26.1" +calico_ctl_version: "{{ calico_version }}" +calico_cni_version: "{{ calico_version }}" +calico_flexvol_version: "{{ calico_version }}" +calico_policy_version: "{{ calico_version }}" +calico_typha_version: "{{ calico_version }}" +calico_apiserver_version: "{{ calico_version }}" +typha_enabled: false +calico_apiserver_enabled: false + +flannel_version: "v0.22.0" +flannel_cni_version: "v1.1.2" +cni_version: "v1.3.0" +weave_version: 2.8.1 +pod_infra_version: "3.9" + +cilium_version: "v1.13.4" +cilium_cli_version: "v0.15.0" +cilium_enable_hubble: false + +kube_ovn_version: "v1.11.5" +kube_ovn_dpdk_version: "19.11-{{ kube_ovn_version }}" +kube_router_version: "v1.6.0" +multus_version: "v3.8" +helm_version: "v3.12.3" +nerdctl_version: "1.6.0" +krew_version: "v0.4.4" +skopeo_version: "v1.13.2" + +# Get kubernetes major version (i.e. 1.17.4 => 1.17) +kube_major_version: "{{ kube_version | regex_replace('^v([0-9])+\\.([0-9]+)\\.[0-9]+', 'v\\1.\\2') }}" + +etcd_supported_versions: + v1.28: "v3.5.9" + v1.27: "v3.5.9" + v1.26: "v3.5.9" +etcd_version: "{{ etcd_supported_versions[kube_major_version] }}" + +crictl_supported_versions: + v1.28: "v1.28.0" + v1.27: "v1.27.1" + v1.26: "v1.26.0" +crictl_version: "{{ crictl_supported_versions[kube_major_version] }}" + +crio_supported_versions: + v1.28: v1.28.1 + v1.27: v1.27.1 + v1.26: v1.26.3 +crio_version: "{{ crio_supported_versions[kube_major_version] }}" + +yq_version: "v4.35.1" + +# Download URLs +kubelet_download_url: "https://dl.k8s.io/release/{{ kube_version }}/bin/linux/{{ image_arch }}/kubelet" +kubectl_download_url: "https://dl.k8s.io/release/{{ kube_version }}/bin/linux/{{ image_arch }}/kubectl" +kubeadm_download_url: "https://dl.k8s.io/release/{{ kubeadm_version }}/bin/linux/{{ image_arch }}/kubeadm" +etcd_download_url: "https://github.com/etcd-io/etcd/releases/download/{{ etcd_version }}/etcd-{{ etcd_version }}-linux-{{ image_arch }}.tar.gz" +cni_download_url: "https://github.com/containernetworking/plugins/releases/download/{{ cni_version }}/cni-plugins-linux-{{ image_arch }}-{{ cni_version }}.tgz" +calicoctl_download_url: "https://github.com/projectcalico/calico/releases/download/{{ calico_ctl_version }}/calicoctl-linux-{{ image_arch }}" +calico_crds_download_url: "https://github.com/projectcalico/calico/archive/{{ calico_version }}.tar.gz" +ciliumcli_download_url: "https://github.com/cilium/cilium-cli/releases/download/{{ cilium_cli_version }}/cilium-linux-{{ image_arch }}.tar.gz" +crictl_download_url: "https://github.com/kubernetes-sigs/cri-tools/releases/download/{{ crictl_version }}/crictl-{{ crictl_version }}-{{ ansible_system | lower }}-{{ image_arch }}.tar.gz" +crio_download_url: "https://storage.googleapis.com/cri-o/artifacts/cri-o.{{ image_arch }}.{{ crio_version }}.tar.gz" +helm_download_url: "https://get.helm.sh/helm-{{ helm_version }}-linux-{{ image_arch }}.tar.gz" +runc_download_url: "https://github.com/opencontainers/runc/releases/download/{{ runc_version }}/runc.{{ image_arch }}" +crun_download_url: "https://github.com/containers/crun/releases/download/{{ crun_version }}/crun-{{ crun_version }}-linux-{{ image_arch }}" +youki_download_url: "https://github.com/containers/youki/releases/download/v{{ youki_version }}/youki_{{ youki_version | regex_replace('\\.', '_') }}_linux.tar.gz" +kata_containers_download_url: "https://github.com/kata-containers/kata-containers/releases/download/{{ kata_containers_version }}/kata-static-{{ kata_containers_version }}-{{ ansible_architecture }}.tar.xz" +# gVisor only supports amd64 and uses x86_64 to in the download link +gvisor_runsc_download_url: "https://storage.googleapis.com/gvisor/releases/release/{{ gvisor_version }}/{{ ansible_architecture }}/runsc" +gvisor_containerd_shim_runsc_download_url: "https://storage.googleapis.com/gvisor/releases/release/{{ gvisor_version }}/{{ ansible_architecture }}/containerd-shim-runsc-v1" +nerdctl_download_url: "https://github.com/containerd/nerdctl/releases/download/v{{ nerdctl_version }}/nerdctl-{{ nerdctl_version }}-{{ ansible_system | lower }}-{{ image_arch }}.tar.gz" +krew_download_url: "https://github.com/kubernetes-sigs/krew/releases/download/{{ krew_version }}/krew-{{ host_os }}_{{ image_arch }}.tar.gz" +containerd_download_url: "https://github.com/containerd/containerd/releases/download/v{{ containerd_version }}/containerd-{{ containerd_version }}-linux-{{ image_arch }}.tar.gz" +cri_dockerd_download_url: "https://github.com/Mirantis/cri-dockerd/releases/download/v{{ cri_dockerd_version }}/cri-dockerd-{{ cri_dockerd_version }}.{{ image_arch }}.tgz" +skopeo_download_url: "https://github.com/lework/skopeo-binary/releases/download/{{ skopeo_version }}/skopeo-linux-{{ image_arch }}" +yq_download_url: "https://github.com/mikefarah/yq/releases/download/{{ yq_version }}/yq_linux_{{ image_arch }}" + +etcd_binary_checksum: "{{ etcd_binary_checksums[image_arch][etcd_version] }}" +cni_binary_checksum: "{{ cni_binary_checksums[image_arch][cni_version] }}" +kubelet_binary_checksum: "{{ kubelet_checksums[image_arch][kube_version] }}" +kubectl_binary_checksum: "{{ kubectl_checksums[image_arch][kube_version] }}" +kubeadm_binary_checksum: "{{ kubeadm_checksums[image_arch][kubeadm_version] }}" +yq_binary_checksum: "{{ yq_checksums[image_arch][yq_version] }}" +calicoctl_binary_checksum: "{{ calicoctl_binary_checksums[image_arch][calico_ctl_version] }}" +calico_crds_archive_checksum: "{{ calico_crds_archive_checksums[calico_version] }}" +ciliumcli_binary_checksum: "{{ ciliumcli_binary_checksums[image_arch][cilium_cli_version] }}" +crictl_binary_checksum: "{{ crictl_checksums[image_arch][crictl_version] }}" +crio_archive_checksum: "{{ crio_archive_checksums[image_arch][crio_version] }}" +cri_dockerd_archive_checksum: "{{ cri_dockerd_archive_checksums[image_arch][cri_dockerd_version] }}" +helm_archive_checksum: "{{ helm_archive_checksums[image_arch][helm_version] }}" +runc_binary_checksum: "{{ runc_checksums[image_arch][runc_version] }}" +crun_binary_checksum: "{{ crun_checksums[image_arch][crun_version] }}" +youki_archive_checksum: "{{ youki_checksums[image_arch][youki_version] }}" +kata_containers_binary_checksum: "{{ kata_containers_binary_checksums[image_arch][kata_containers_version] }}" +gvisor_runsc_binary_checksum: "{{ gvisor_runsc_binary_checksums[image_arch][gvisor_version] }}" +gvisor_containerd_shim_binary_checksum: "{{ gvisor_containerd_shim_binary_checksums[image_arch][gvisor_version] }}" +nerdctl_archive_checksum: "{{ nerdctl_archive_checksums[image_arch][nerdctl_version] }}" +krew_archive_checksum: "{{ krew_archive_checksums[host_os][image_arch][krew_version] }}" +containerd_archive_checksum: "{{ containerd_archive_checksums[image_arch][containerd_version] }}" +skopeo_binary_checksum: "{{ skopeo_binary_checksums[image_arch][skopeo_version] }}" + +# Containers +# In some cases, we need a way to set --registry-mirror or --insecure-registry for docker, +# it helps a lot for local private development or bare metal environment. +# So you need define --registry-mirror or --insecure-registry, and modify the following url address. +# example: +# You need to deploy kubernetes cluster on local private development. +# Also provide the address of your own private registry. +# And use --insecure-registry options for docker +kube_proxy_image_repo: "{{ kube_image_repo }}/kube-proxy" +etcd_image_repo: "{{ quay_image_repo }}/coreos/etcd" +etcd_image_tag: "{{ etcd_version }}" +flannel_image_repo: "{{ docker_image_repo }}/flannel/flannel" +flannel_image_tag: "{{ flannel_version }}" +flannel_init_image_repo: "{{ docker_image_repo }}/flannel/flannel-cni-plugin" +flannel_init_image_tag: "{{ flannel_cni_version }}" +calico_node_image_repo: "{{ quay_image_repo }}/calico/node" +calico_node_image_tag: "{{ calico_version }}" +calico_cni_image_repo: "{{ quay_image_repo }}/calico/cni" +calico_cni_image_tag: "{{ calico_cni_version }}" +calico_flexvol_image_repo: "{{ quay_image_repo }}/calico/pod2daemon-flexvol" +calico_flexvol_image_tag: "{{ calico_flexvol_version }}" +calico_policy_image_repo: "{{ quay_image_repo }}/calico/kube-controllers" +calico_policy_image_tag: "{{ calico_policy_version }}" +calico_typha_image_repo: "{{ quay_image_repo }}/calico/typha" +calico_typha_image_tag: "{{ calico_typha_version }}" +calico_apiserver_image_repo: "{{ quay_image_repo }}/calico/apiserver" +calico_apiserver_image_tag: "{{ calico_apiserver_version }}" +pod_infra_image_repo: "{{ kube_image_repo }}/pause" +pod_infra_image_tag: "{{ pod_infra_version }}" +netcheck_version: "v1.2.2" +netcheck_agent_image_repo: "{{ docker_image_repo }}/mirantis/k8s-netchecker-agent" +netcheck_agent_image_tag: "{{ netcheck_version }}" +netcheck_server_image_repo: "{{ docker_image_repo }}/mirantis/k8s-netchecker-server" +netcheck_server_image_tag: "{{ netcheck_version }}" +netcheck_etcd_image_tag: "v3.4.17" +weave_kube_image_repo: "{{ docker_image_repo }}/weaveworks/weave-kube" +weave_kube_image_tag: "{{ weave_version }}" +weave_npc_image_repo: "{{ docker_image_repo }}/weaveworks/weave-npc" +weave_npc_image_tag: "{{ weave_version }}" +cilium_image_repo: "{{ quay_image_repo }}/cilium/cilium" +cilium_image_tag: "{{ cilium_version }}" +cilium_operator_image_repo: "{{ quay_image_repo }}/cilium/operator" +cilium_operator_image_tag: "{{ cilium_version }}" +cilium_hubble_relay_image_repo: "{{ quay_image_repo }}/cilium/hubble-relay" +cilium_hubble_relay_image_tag: "{{ cilium_version }}" +cilium_hubble_certgen_image_repo: "{{ quay_image_repo }}/cilium/certgen" +cilium_hubble_certgen_image_tag: "v0.1.8" +cilium_hubble_ui_image_repo: "{{ quay_image_repo }}/cilium/hubble-ui" +cilium_hubble_ui_image_tag: "v0.11.0" +cilium_hubble_ui_backend_image_repo: "{{ quay_image_repo }}/cilium/hubble-ui-backend" +cilium_hubble_ui_backend_image_tag: "v0.11.0" +cilium_hubble_envoy_image_repo: "{{ docker_image_repo }}/envoyproxy/envoy" +cilium_hubble_envoy_image_tag: "v1.22.5" +kube_ovn_container_image_repo: "{{ docker_image_repo }}/kubeovn/kube-ovn" +kube_ovn_container_image_tag: "{{ kube_ovn_version }}" +kube_ovn_dpdk_container_image_repo: "{{ docker_image_repo }}/kubeovn/kube-ovn-dpdk" +kube_ovn_dpdk_container_image_tag: "{{ kube_ovn_dpdk_version }}" +kube_router_image_repo: "{{ docker_image_repo }}/cloudnativelabs/kube-router" +kube_router_image_tag: "{{ kube_router_version }}" +multus_image_repo: "{{ github_image_repo }}/k8snetworkplumbingwg/multus-cni" +multus_image_tag: "{{ multus_version }}" + +kube_vip_image_repo: "{{ github_image_repo }}/kube-vip/kube-vip" +kube_vip_image_tag: v0.5.12 +nginx_image_repo: "{{ docker_image_repo }}/library/nginx" +nginx_image_tag: 1.25.2-alpine +haproxy_image_repo: "{{ docker_image_repo }}/library/haproxy" +haproxy_image_tag: 2.8.2-alpine + +# Coredns version should be supported by corefile-migration (or at least work with) +# bundle with kubeadm; if not 'basic' upgrade can sometimes fail + +coredns_version: "{{ 'v1.10.1' if (kube_version is version('v1.27.0', '>=')) else 'v1.9.3' }}" +coredns_image_is_namespaced: "{{ (coredns_version is version('v1.7.1', '>=')) }}" + +coredns_image_repo: "{{ kube_image_repo }}{{ '/coredns/coredns' if (coredns_image_is_namespaced | bool) else '/coredns' }}" +coredns_image_tag: "{{ coredns_version if (coredns_image_is_namespaced | bool) else (coredns_version | regex_replace('^v', '')) }}" + +nodelocaldns_version: "1.22.20" +nodelocaldns_image_repo: "{{ kube_image_repo }}/dns/k8s-dns-node-cache" +nodelocaldns_image_tag: "{{ nodelocaldns_version }}" + +dnsautoscaler_version: v1.8.8 +dnsautoscaler_image_repo: "{{ kube_image_repo }}/cpa/cluster-proportional-autoscaler" +dnsautoscaler_image_tag: "{{ dnsautoscaler_version }}" + +registry_version: "2.8.1" +registry_image_repo: "{{ docker_image_repo }}/library/registry" +registry_image_tag: "{{ registry_version }}" +metrics_server_version: "v0.6.4" +metrics_server_image_repo: "{{ kube_image_repo }}/metrics-server/metrics-server" +metrics_server_image_tag: "{{ metrics_server_version }}" +local_volume_provisioner_version: "v2.5.0" +local_volume_provisioner_image_repo: "{{ kube_image_repo }}/sig-storage/local-volume-provisioner" +local_volume_provisioner_image_tag: "{{ local_volume_provisioner_version }}" +cephfs_provisioner_version: "v2.1.0-k8s1.11" +cephfs_provisioner_image_repo: "{{ quay_image_repo }}/external_storage/cephfs-provisioner" +cephfs_provisioner_image_tag: "{{ cephfs_provisioner_version }}" +rbd_provisioner_version: "v2.1.1-k8s1.11" +rbd_provisioner_image_repo: "{{ quay_image_repo }}/external_storage/rbd-provisioner" +rbd_provisioner_image_tag: "{{ rbd_provisioner_version }}" +local_path_provisioner_version: "v0.0.24" +local_path_provisioner_image_repo: "{{ docker_image_repo }}/rancher/local-path-provisioner" +local_path_provisioner_image_tag: "{{ local_path_provisioner_version }}" +ingress_nginx_version: "v1.8.2" +ingress_nginx_controller_image_repo: "{{ kube_image_repo }}/ingress-nginx/controller" +ingress_nginx_controller_image_tag: "{{ ingress_nginx_version }}" +ingress_nginx_kube_webhook_certgen_image_repo: "{{ kube_image_repo }}/ingress-nginx/kube-webhook-certgen" +ingress_nginx_kube_webhook_certgen_image_tag: "v20230407" +alb_ingress_image_repo: "{{ docker_image_repo }}/amazon/aws-alb-ingress-controller" +alb_ingress_image_tag: "v1.1.9" +cert_manager_version: "v1.11.1" +cert_manager_controller_image_repo: "{{ quay_image_repo }}/jetstack/cert-manager-controller" +cert_manager_controller_image_tag: "{{ cert_manager_version }}" +cert_manager_cainjector_image_repo: "{{ quay_image_repo }}/jetstack/cert-manager-cainjector" +cert_manager_cainjector_image_tag: "{{ cert_manager_version }}" +cert_manager_webhook_image_repo: "{{ quay_image_repo }}/jetstack/cert-manager-webhook" +cert_manager_webhook_image_tag: "{{ cert_manager_version }}" + +csi_attacher_image_repo: "{{ kube_image_repo }}/sig-storage/csi-attacher" +csi_attacher_image_tag: "v3.3.0" +csi_provisioner_image_repo: "{{ kube_image_repo }}/sig-storage/csi-provisioner" +csi_provisioner_image_tag: "v3.0.0" +csi_snapshotter_image_repo: "{{ kube_image_repo }}/sig-storage/csi-snapshotter" +csi_snapshotter_image_tag: "v5.0.0" +csi_resizer_image_repo: "{{ kube_image_repo }}/sig-storage/csi-resizer" +csi_resizer_image_tag: "v1.3.0" +csi_node_driver_registrar_image_repo: "{{ kube_image_repo }}/sig-storage/csi-node-driver-registrar" +csi_node_driver_registrar_image_tag: "v2.4.0" +csi_livenessprobe_image_repo: "{{ kube_image_repo }}/sig-storage/livenessprobe" +csi_livenessprobe_image_tag: "v2.5.0" + +snapshot_controller_supported_versions: + v1.28: "v4.2.1" + v1.27: "v4.2.1" + v1.26: "v4.2.1" +snapshot_controller_image_repo: "{{ kube_image_repo }}/sig-storage/snapshot-controller" +snapshot_controller_image_tag: "{{ snapshot_controller_supported_versions[kube_major_version] }}" + +cinder_csi_plugin_version: "v1.22.0" +cinder_csi_plugin_image_repo: "{{ docker_image_repo }}/k8scloudprovider/cinder-csi-plugin" +cinder_csi_plugin_image_tag: "{{ cinder_csi_plugin_version }}" + +aws_ebs_csi_plugin_version: "v0.5.0" +aws_ebs_csi_plugin_image_repo: "{{ docker_image_repo }}/amazon/aws-ebs-csi-driver" +aws_ebs_csi_plugin_image_tag: "{{ aws_ebs_csi_plugin_version }}" + +gcp_pd_csi_plugin_version: "v1.9.2" +gcp_pd_csi_plugin_image_repo: "{{ kube_image_repo }}/cloud-provider-gcp/gcp-compute-persistent-disk-csi-driver" +gcp_pd_csi_plugin_image_tag: "{{ gcp_pd_csi_plugin_version }}" + +azure_csi_image_repo: "mcr.microsoft.com/oss/kubernetes-csi" +azure_csi_provisioner_image_tag: "v2.2.2" +azure_csi_attacher_image_tag: "v3.3.0" +azure_csi_resizer_image_tag: "v1.3.0" +azure_csi_livenessprobe_image_tag: "v2.5.0" +azure_csi_node_registrar_image_tag: "v2.4.0" +azure_csi_snapshotter_image_tag: "v3.0.3" +azure_csi_plugin_version: "v1.10.0" +azure_csi_plugin_image_repo: "mcr.microsoft.com/k8s/csi" +azure_csi_plugin_image_tag: "{{ azure_csi_plugin_version }}" + +gcp_pd_csi_image_repo: "gke.gcr.io" +gcp_pd_csi_driver_image_tag: "v0.7.0-gke.0" +gcp_pd_csi_provisioner_image_tag: "v1.5.0-gke.0" +gcp_pd_csi_attacher_image_tag: "v2.1.1-gke.0" +gcp_pd_csi_resizer_image_tag: "v0.4.0-gke.0" +gcp_pd_csi_registrar_image_tag: "v1.2.0-gke.0" + +dashboard_image_repo: "{{ docker_image_repo }}/kubernetesui/dashboard" +dashboard_image_tag: "v2.7.0" +dashboard_metrics_scraper_repo: "{{ docker_image_repo }}/kubernetesui/metrics-scraper" +dashboard_metrics_scraper_tag: "v1.0.8" + +metallb_speaker_image_repo: "{{ quay_image_repo }}/metallb/speaker" +metallb_controller_image_repo: "{{ quay_image_repo }}/metallb/controller" +metallb_version: v0.13.9 + +downloads: + netcheck_server: + enabled: "{{ deploy_netchecker }}" + container: true + repo: "{{ netcheck_server_image_repo }}" + tag: "{{ netcheck_server_image_tag }}" + sha256: "{{ netcheck_server_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + netcheck_agent: + enabled: "{{ deploy_netchecker }}" + container: true + repo: "{{ netcheck_agent_image_repo }}" + tag: "{{ netcheck_agent_image_tag }}" + sha256: "{{ netcheck_agent_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + etcd: + container: "{{ etcd_deployment_type != 'host' }}" + file: "{{ etcd_deployment_type == 'host' }}" + enabled: true + version: "{{ etcd_version }}" + dest: "{{ local_release_dir }}/etcd-{{ etcd_version }}-linux-{{ image_arch }}.tar.gz" + repo: "{{ etcd_image_repo }}" + tag: "{{ etcd_image_tag }}" + sha256: >- + {{ etcd_binary_checksum if (etcd_deployment_type == 'host') + else etcd_digest_checksum | d(None) }} + url: "{{ etcd_download_url }}" + unarchive: "{{ etcd_deployment_type == 'host' }}" + owner: "root" + mode: "0755" + groups: + - etcd + + cni: + enabled: true + file: true + version: "{{ cni_version }}" + dest: "{{ local_release_dir }}/cni-plugins-linux-{{ image_arch }}-{{ cni_version }}.tgz" + sha256: "{{ cni_binary_checksum }}" + url: "{{ cni_download_url }}" + unarchive: false + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + kubeadm: + enabled: true + file: true + version: "{{ kubeadm_version }}" + dest: "{{ local_release_dir }}/kubeadm-{{ kubeadm_version }}-{{ image_arch }}" + sha256: "{{ kubeadm_binary_checksum }}" + url: "{{ kubeadm_download_url }}" + unarchive: false + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + kubelet: + enabled: true + file: true + version: "{{ kube_version }}" + dest: "{{ local_release_dir }}/kubelet-{{ kube_version }}-{{ image_arch }}" + sha256: "{{ kubelet_binary_checksum }}" + url: "{{ kubelet_download_url }}" + unarchive: false + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + kubectl: + enabled: true + file: true + version: "{{ kube_version }}" + dest: "{{ local_release_dir }}/kubectl-{{ kube_version }}-{{ image_arch }}" + sha256: "{{ kubectl_binary_checksum }}" + url: "{{ kubectl_download_url }}" + unarchive: false + owner: "root" + mode: "0755" + groups: + - kube_control_plane + + crictl: + file: true + enabled: true + version: "{{ crictl_version }}" + dest: "{{ local_release_dir }}/crictl-{{ crictl_version }}-linux-{{ image_arch }}.tar.gz" + sha256: "{{ crictl_binary_checksum }}" + url: "{{ crictl_download_url }}" + unarchive: true + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + crio: + file: true + enabled: "{{ container_manager == 'crio' }}" + version: "{{ crio_version }}" + dest: "{{ local_release_dir }}/cri-o.{{ image_arch }}.{{ crio_version }}tar.gz" + sha256: "{{ crio_archive_checksum }}" + url: "{{ crio_download_url }}" + unarchive: true + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + cri_dockerd: + file: true + enabled: "{{ container_manager == 'docker' }}" + version: "{{ cri_dockerd_version }}" + dest: "{{ local_release_dir }}/cri-dockerd-{{ cri_dockerd_version }}.{{ image_arch }}.tar.gz" + sha256: "{{ cri_dockerd_archive_checksum }}" + url: "{{ cri_dockerd_download_url }}" + unarchive: true + unarchive_extra_opts: + - --strip=1 + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + crun: + file: true + enabled: "{{ crun_enabled }}" + version: "{{ crun_version }}" + dest: "{{ local_release_dir }}/crun-{{ crun_version }}-{{ image_arch }}" + sha256: "{{ crun_binary_checksum }}" + url: "{{ crun_download_url }}" + unarchive: false + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + youki: + file: true + enabled: "{{ youki_enabled }}" + version: "{{ youki_version }}" + dest: "{{ local_release_dir }}/youki_{{ youki_version | regex_replace('\\.', '_') }}_linux.tar.gz" + sha256: "{{ youki_archive_checksum }}" + url: "{{ youki_download_url }}" + unarchive: true + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + runc: + file: true + enabled: "{{ container_manager == 'containerd' }}" + version: "{{ runc_version }}" + dest: "{{ local_release_dir }}/runc-{{ runc_version }}.{{ image_arch }}" + sha256: "{{ runc_binary_checksum }}" + url: "{{ runc_download_url }}" + unarchive: false + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + kata_containers: + enabled: "{{ kata_containers_enabled }}" + file: true + version: "{{ kata_containers_version }}" + dest: "{{ local_release_dir }}/kata-static-{{ kata_containers_version }}-{{ image_arch }}.tar.xz" + sha256: "{{ kata_containers_binary_checksum }}" + url: "{{ kata_containers_download_url }}" + unarchive: false + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + containerd: + enabled: "{{ container_manager == 'containerd' }}" + file: true + version: "{{ containerd_version }}" + dest: "{{ local_release_dir }}/containerd-{{ containerd_version }}-linux-{{ image_arch }}.tar.gz" + sha256: "{{ containerd_archive_checksum }}" + url: "{{ containerd_download_url }}" + unarchive: false + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + gvisor_runsc: + enabled: "{{ gvisor_enabled }}" + file: true + version: "{{ gvisor_version }}" + dest: "{{ local_release_dir }}/gvisor-runsc-{{ gvisor_version }}-{{ ansible_architecture }}" + sha256: "{{ gvisor_runsc_binary_checksum }}" + url: "{{ gvisor_runsc_download_url }}" + unarchive: false + owner: "root" + mode: 755 + groups: + - k8s_cluster + + gvisor_containerd_shim: + enabled: "{{ gvisor_enabled }}" + file: true + version: "{{ gvisor_version }}" + dest: "{{ local_release_dir }}/gvisor-containerd-shim-runsc-v1-{{ gvisor_version }}-{{ ansible_architecture }}" + sha256: "{{ gvisor_containerd_shim_binary_checksum }}" + url: "{{ gvisor_containerd_shim_runsc_download_url }}" + unarchive: false + owner: "root" + mode: 755 + groups: + - k8s_cluster + + nerdctl: + file: true + enabled: "{{ container_manager == 'containerd' }}" + version: "{{ nerdctl_version }}" + dest: "{{ local_release_dir }}/nerdctl-{{ nerdctl_version }}-linux-{{ image_arch }}.tar.gz" + sha256: "{{ nerdctl_archive_checksum }}" + url: "{{ nerdctl_download_url }}" + unarchive: true + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + skopeo: + file: true + enabled: "{{ container_manager == 'crio' }}" + version: "{{ skopeo_version }}" + dest: "{{ local_release_dir }}/skopeo-{{ skopeo_version }}-{{ image_arch }}" + sha256: "{{ skopeo_binary_checksum }}" + url: "{{ skopeo_download_url }}" + unarchive: false + owner: "root" + mode: "0755" + groups: + - kube_control_plane + + cilium: + enabled: "{{ kube_network_plugin == 'cilium' or cilium_deploy_additionally | default(false) | bool }}" + container: true + repo: "{{ cilium_image_repo }}" + tag: "{{ cilium_image_tag }}" + sha256: "{{ cilium_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + cilium_operator: + enabled: "{{ kube_network_plugin == 'cilium' or cilium_deploy_additionally | default(false) | bool }}" + container: true + repo: "{{ cilium_operator_image_repo }}" + tag: "{{ cilium_operator_image_tag }}" + sha256: "{{ cilium_operator_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + cilium_hubble_relay: + enabled: "{{ cilium_enable_hubble }}" + container: true + repo: "{{ cilium_hubble_relay_image_repo }}" + tag: "{{ cilium_hubble_relay_image_tag }}" + sha256: "{{ cilium_hubble_relay_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + cilium_hubble_certgen: + enabled: "{{ cilium_enable_hubble }}" + container: true + repo: "{{ cilium_hubble_certgen_image_repo }}" + tag: "{{ cilium_hubble_certgen_image_tag }}" + sha256: "{{ cilium_hubble_certgen_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + cilium_hubble_ui: + enabled: "{{ cilium_enable_hubble }}" + container: true + repo: "{{ cilium_hubble_ui_image_repo }}" + tag: "{{ cilium_hubble_ui_image_tag }}" + sha256: "{{ cilium_hubble_ui_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + cilium_hubble_ui_backend: + enabled: "{{ cilium_enable_hubble }}" + container: true + repo: "{{ cilium_hubble_ui_backend_image_repo }}" + tag: "{{ cilium_hubble_ui_backend_image_tag }}" + sha256: "{{ cilium_hubble_ui_backend_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + cilium_hubble_envoy: + enabled: "{{ cilium_enable_hubble }}" + container: true + repo: "{{ cilium_hubble_envoy_image_repo }}" + tag: "{{ cilium_hubble_envoy_image_tag }}" + sha256: "{{ cilium_hubble_envoy_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + ciliumcli: + enabled: "{{ kube_network_plugin == 'cilium' or cilium_deploy_additionally | default(false) | bool }}" + file: true + version: "{{ cilium_cli_version }}" + dest: "{{ local_release_dir }}/cilium-{{ cilium_cli_version }}-{{ image_arch }}" + sha256: "{{ ciliumcli_binary_checksum }}" + url: "{{ ciliumcli_download_url }}" + unarchive: true + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + multus: + enabled: "{{ kube_network_plugin_multus }}" + container: true + repo: "{{ multus_image_repo }}" + tag: "{{ multus_image_tag }}" + sha256: "{{ multus_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + flannel: + enabled: "{{ kube_network_plugin == 'flannel' }}" + container: true + repo: "{{ flannel_image_repo }}" + tag: "{{ flannel_image_tag }}" + sha256: "{{ flannel_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + flannel_init: + enabled: "{{ kube_network_plugin == 'flannel' }}" + container: true + repo: "{{ flannel_init_image_repo }}" + tag: "{{ flannel_init_image_tag }}" + sha256: "{{ flannel_init_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + calicoctl: + enabled: "{{ kube_network_plugin == 'calico' }}" + file: true + version: "{{ calico_ctl_version }}" + dest: "{{ local_release_dir }}/calicoctl-{{ calico_ctl_version }}-{{ image_arch }}" + sha256: "{{ calicoctl_binary_checksum }}" + url: "{{ calicoctl_download_url }}" + unarchive: false + owner: "root" + mode: "0755" + groups: + - k8s_cluster + + calico_node: + enabled: "{{ kube_network_plugin == 'calico' }}" + container: true + repo: "{{ calico_node_image_repo }}" + tag: "{{ calico_node_image_tag }}" + sha256: "{{ calico_node_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + calico_cni: + enabled: "{{ kube_network_plugin == 'calico' }}" + container: true + repo: "{{ calico_cni_image_repo }}" + tag: "{{ calico_cni_image_tag }}" + sha256: "{{ calico_cni_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + calico_flexvol: + enabled: "{{ kube_network_plugin == 'calico' }}" + container: true + repo: "{{ calico_flexvol_image_repo }}" + tag: "{{ calico_flexvol_image_tag }}" + sha256: "{{ calico_flexvol_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + calico_policy: + enabled: "{{ enable_network_policy and kube_network_plugin in ['calico'] }}" + container: true + repo: "{{ calico_policy_image_repo }}" + tag: "{{ calico_policy_image_tag }}" + sha256: "{{ calico_policy_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + calico_typha: + enabled: "{{ typha_enabled }}" + container: true + repo: "{{ calico_typha_image_repo }}" + tag: "{{ calico_typha_image_tag }}" + sha256: "{{ calico_typha_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + calico_apiserver: + enabled: "{{ calico_apiserver_enabled }}" + container: true + repo: "{{ calico_apiserver_image_repo }}" + tag: "{{ calico_apiserver_image_tag }}" + sha256: "{{ calico_apiserver_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + calico_crds: + file: true + enabled: "{{ kube_network_plugin == 'calico' and calico_datastore == 'kdd' }}" + version: "{{ calico_version }}" + dest: "{{ local_release_dir }}/calico-{{ calico_version }}-kdd-crds/{{ calico_version }}.tar.gz" + sha256: "{{ calico_crds_archive_checksum }}" + url: "{{ calico_crds_download_url }}" + unarchive: true + unarchive_extra_opts: + - "{{ '--strip=6' if (calico_version is version('v3.22.3', '<')) else '--strip=3' }}" + - "--wildcards" + - "{{ '*/_includes/charts/calico/crds/kdd/' if (calico_version is version('v3.22.3', '<')) else '*/libcalico-go/config/crd/' }}" + owner: "root" + mode: "0755" + groups: + - kube_control_plane + + weave_kube: + enabled: "{{ kube_network_plugin == 'weave' }}" + container: true + repo: "{{ weave_kube_image_repo }}" + tag: "{{ weave_kube_image_tag }}" + sha256: "{{ weave_kube_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + weave_npc: + enabled: "{{ kube_network_plugin == 'weave' }}" + container: true + repo: "{{ weave_npc_image_repo }}" + tag: "{{ weave_npc_image_tag }}" + sha256: "{{ weave_npc_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + kube_ovn: + enabled: "{{ kube_network_plugin == 'kube-ovn' }}" + container: true + repo: "{{ kube_ovn_container_image_repo }}" + tag: "{{ kube_ovn_container_image_tag }}" + sha256: "{{ kube_ovn_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + kube_router: + enabled: "{{ kube_network_plugin == 'kube-router' }}" + container: true + repo: "{{ kube_router_image_repo }}" + tag: "{{ kube_router_image_tag }}" + sha256: "{{ kube_router_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + pod_infra: + enabled: true + container: true + repo: "{{ pod_infra_image_repo }}" + tag: "{{ pod_infra_image_tag }}" + sha256: "{{ pod_infra_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + kube-vip: + enabled: "{{ kube_vip_enabled }}" + container: true + repo: "{{ kube_vip_image_repo }}" + tag: "{{ kube_vip_image_tag }}" + sha256: "{{ kube_vip_digest_checksum | default(None) }}" + groups: + - kube_control_plane + + nginx: + enabled: "{{ loadbalancer_apiserver_localhost and loadbalancer_apiserver_type == 'nginx' }}" + container: true + repo: "{{ nginx_image_repo }}" + tag: "{{ nginx_image_tag }}" + sha256: "{{ nginx_digest_checksum | default(None) }}" + groups: + - kube_node + + haproxy: + enabled: "{{ loadbalancer_apiserver_localhost and loadbalancer_apiserver_type == 'haproxy' }}" + container: true + repo: "{{ haproxy_image_repo }}" + tag: "{{ haproxy_image_tag }}" + sha256: "{{ haproxy_digest_checksum | default(None) }}" + groups: + - kube_node + + coredns: + enabled: "{{ dns_mode in ['coredns', 'coredns_dual'] }}" + container: true + repo: "{{ coredns_image_repo }}" + tag: "{{ coredns_image_tag }}" + sha256: "{{ coredns_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + nodelocaldns: + enabled: "{{ enable_nodelocaldns }}" + container: true + repo: "{{ nodelocaldns_image_repo }}" + tag: "{{ nodelocaldns_image_tag }}" + sha256: "{{ nodelocaldns_digest_checksum | default(None) }}" + groups: + - k8s_cluster + + dnsautoscaler: + enabled: "{{ dns_mode in ['coredns', 'coredns_dual'] }}" + container: true + repo: "{{ dnsautoscaler_image_repo }}" + tag: "{{ dnsautoscaler_image_tag }}" + sha256: "{{ dnsautoscaler_digest_checksum | default(None) }}" + groups: + - kube_control_plane + + helm: + enabled: "{{ helm_enabled }}" + file: true + version: "{{ helm_version }}" + dest: "{{ local_release_dir }}/helm-{{ helm_version }}/helm-{{ helm_version }}-linux-{{ image_arch }}.tar.gz" + sha256: "{{ helm_archive_checksum }}" + url: "{{ helm_download_url }}" + unarchive: true + owner: "root" + mode: "0755" + groups: + - kube_control_plane + + krew: + enabled: "{{ krew_enabled }}" + file: true + version: "{{ krew_version }}" + dest: "{{ local_release_dir }}/krew-{{ host_os }}_{{ image_arch }}.tar.gz" + sha256: "{{ krew_archive_checksum }}" + url: "{{ krew_download_url }}" + unarchive: true + owner: "root" + mode: "0755" + groups: + - kube_control_plane + + registry: + enabled: "{{ registry_enabled }}" + container: true + repo: "{{ registry_image_repo }}" + tag: "{{ registry_image_tag }}" + sha256: "{{ registry_digest_checksum | default(None) }}" + groups: + - kube_node + + metrics_server: + enabled: "{{ metrics_server_enabled }}" + container: true + repo: "{{ metrics_server_image_repo }}" + tag: "{{ metrics_server_image_tag }}" + sha256: "{{ metrics_server_digest_checksum | default(None) }}" + groups: + - kube_control_plane + + local_volume_provisioner: + enabled: "{{ local_volume_provisioner_enabled }}" + container: true + repo: "{{ local_volume_provisioner_image_repo }}" + tag: "{{ local_volume_provisioner_image_tag }}" + sha256: "{{ local_volume_provisioner_digest_checksum | default(None) }}" + groups: + - kube_node + + cephfs_provisioner: + enabled: "{{ cephfs_provisioner_enabled }}" + container: true + repo: "{{ cephfs_provisioner_image_repo }}" + tag: "{{ cephfs_provisioner_image_tag }}" + sha256: "{{ cephfs_provisioner_digest_checksum | default(None) }}" + groups: + - kube_node + + rbd_provisioner: + enabled: "{{ rbd_provisioner_enabled }}" + container: true + repo: "{{ rbd_provisioner_image_repo }}" + tag: "{{ rbd_provisioner_image_tag }}" + sha256: "{{ rbd_provisioner_digest_checksum | default(None) }}" + groups: + - kube_node + + local_path_provisioner: + enabled: "{{ local_path_provisioner_enabled }}" + container: true + repo: "{{ local_path_provisioner_image_repo }}" + tag: "{{ local_path_provisioner_image_tag }}" + sha256: "{{ local_path_provisioner_digest_checksum | default(None) }}" + groups: + - kube_node + + ingress_nginx_controller: + enabled: "{{ ingress_nginx_enabled }}" + container: true + repo: "{{ ingress_nginx_controller_image_repo }}" + tag: "{{ ingress_nginx_controller_image_tag }}" + sha256: "{{ ingress_nginx_controller_digest_checksum | default(None) }}" + groups: + - kube_node + + ingress_alb_controller: + enabled: "{{ ingress_alb_enabled }}" + container: true + repo: "{{ alb_ingress_image_repo }}" + tag: "{{ alb_ingress_image_tag }}" + sha256: "{{ ingress_alb_controller_digest_checksum | default(None) }}" + groups: + - kube_node + + cert_manager_controller: + enabled: "{{ cert_manager_enabled }}" + container: true + repo: "{{ cert_manager_controller_image_repo }}" + tag: "{{ cert_manager_controller_image_tag }}" + sha256: "{{ cert_manager_controller_digest_checksum | default(None) }}" + groups: + - kube_node + + cert_manager_cainjector: + enabled: "{{ cert_manager_enabled }}" + container: true + repo: "{{ cert_manager_cainjector_image_repo }}" + tag: "{{ cert_manager_cainjector_image_tag }}" + sha256: "{{ cert_manager_cainjector_digest_checksum | default(None) }}" + groups: + - kube_node + + cert_manager_webhook: + enabled: "{{ cert_manager_enabled }}" + container: true + repo: "{{ cert_manager_webhook_image_repo }}" + tag: "{{ cert_manager_webhook_image_tag }}" + sha256: "{{ cert_manager_webhook_digest_checksum | default(None) }}" + groups: + - kube_node + + csi_attacher: + enabled: "{{ cinder_csi_enabled or aws_ebs_csi_enabled }}" + container: true + repo: "{{ csi_attacher_image_repo }}" + tag: "{{ csi_attacher_image_tag }}" + sha256: "{{ csi_attacher_digest_checksum | default(None) }}" + groups: + - kube_node + + csi_provisioner: + enabled: "{{ cinder_csi_enabled or aws_ebs_csi_enabled }}" + container: true + repo: "{{ csi_provisioner_image_repo }}" + tag: "{{ csi_provisioner_image_tag }}" + sha256: "{{ csi_provisioner_digest_checksum | default(None) }}" + groups: + - kube_node + + csi_snapshotter: + enabled: "{{ cinder_csi_enabled or aws_ebs_csi_enabled }}" + container: true + repo: "{{ csi_snapshotter_image_repo }}" + tag: "{{ csi_snapshotter_image_tag }}" + sha256: "{{ csi_snapshotter_digest_checksum | default(None) }}" + groups: + - kube_node + + snapshot_controller: + enabled: "{{ csi_snapshot_controller_enabled }}" + container: true + repo: "{{ snapshot_controller_image_repo }}" + tag: "{{ snapshot_controller_image_tag }}" + sha256: "{{ snapshot_controller_digest_checksum | default(None) }}" + groups: + - kube_node + + csi_resizer: + enabled: "{{ cinder_csi_enabled or aws_ebs_csi_enabled }}" + container: true + repo: "{{ csi_resizer_image_repo }}" + tag: "{{ csi_resizer_image_tag }}" + sha256: "{{ csi_resizer_digest_checksum | default(None) }}" + groups: + - kube_node + + csi_node_driver_registrar: + enabled: "{{ cinder_csi_enabled or aws_ebs_csi_enabled }}" + container: true + repo: "{{ csi_node_driver_registrar_image_repo }}" + tag: "{{ csi_node_driver_registrar_image_tag }}" + sha256: "{{ csi_node_driver_registrar_digest_checksum | default(None) }}" + groups: + - kube_node + + cinder_csi_plugin: + enabled: "{{ cinder_csi_enabled }}" + container: true + repo: "{{ cinder_csi_plugin_image_repo }}" + tag: "{{ cinder_csi_plugin_image_tag }}" + sha256: "{{ cinder_csi_plugin_digest_checksum | default(None) }}" + groups: + - kube_node + + aws_ebs_csi_plugin: + enabled: "{{ aws_ebs_csi_enabled }}" + container: true + repo: "{{ aws_ebs_csi_plugin_image_repo }}" + tag: "{{ aws_ebs_csi_plugin_image_tag }}" + sha256: "{{ aws_ebs_csi_plugin_digest_checksum | default(None) }}" + groups: + - kube_node + + dashboard: + enabled: "{{ dashboard_enabled }}" + container: true + repo: "{{ dashboard_image_repo }}" + tag: "{{ dashboard_image_tag }}" + sha256: "{{ dashboard_digest_checksum | default(None) }}" + groups: + - kube_control_plane + + dashboard_metrics_scrapper: + enabled: "{{ dashboard_enabled }}" + container: true + repo: "{{ dashboard_metrics_scraper_repo }}" + tag: "{{ dashboard_metrics_scraper_tag }}" + sha256: "{{ dashboard_digest_checksum | default(None) }}" + groups: + - kube_control_plane + + metallb_speaker: + enabled: "{{ metallb_speaker_enabled }}" + container: true + repo: "{{ metallb_speaker_image_repo }}" + tag: "{{ metallb_version }}" + sha256: "{{ metallb_speaker_digest_checksum | default(None) }}" + groups: + - kube_control_plane + + metallb_controller: + enabled: "{{ metallb_enabled }}" + container: true + repo: "{{ metallb_controller_image_repo }}" + tag: "{{ metallb_version }}" + sha256: "{{ metallb_controller_digest_checksum | default(None) }}" + groups: + - kube_control_plane + + yq: + enabled: "{{ argocd_enabled }}" + file: true + version: "{{ yq_version }}" + dest: "{{ local_release_dir }}/yq-{{ yq_version }}-{{ image_arch }}" + sha256: "{{ yq_binary_checksum | default(None) }}" + url: "{{ yq_download_url }}" + unarchive: false + owner: "root" + mode: "0755" + groups: + - kube_control_plane + +download_defaults: + container: false + file: false + repo: None + tag: None + enabled: false + dest: None + version: None + url: None + unarchive: false + owner: "{{ kube_owner }}" + mode: None diff --git a/kubespray/roles/download/meta/main.yml b/kubespray/roles/download/meta/main.yml new file mode 100644 index 0000000..61d3ffe --- /dev/null +++ b/kubespray/roles/download/meta/main.yml @@ -0,0 +1,2 @@ +--- +allow_duplicates: true diff --git a/kubespray/roles/download/tasks/check_pull_required.yml b/kubespray/roles/download/tasks/check_pull_required.yml new file mode 100644 index 0000000..e5ae1dc --- /dev/null +++ b/kubespray/roles/download/tasks/check_pull_required.yml @@ -0,0 +1,25 @@ +--- +# The image_info_command depends on the Container Runtime and will output something like the following: +# nginx:1.15,gcr.io/google-containers/kube-proxy:v1.14.1,gcr.io/google-containers/kube-proxy@sha256:44af2833c6cbd9a7fc2e9d2f5244a39dfd2e31ad91bf9d4b7d810678db738ee9,gcr.io/google-containers/kube-apiserver:v1.14.1,etc... +- name: Check_pull_required | Generate a list of information about the images on a node # noqa command-instead-of-shell - image_info_command contains a pipe, therefore requiring shell + shell: "{{ image_info_command }}" + register: docker_images + changed_when: false + check_mode: no + when: not download_always_pull + +- name: Check_pull_required | Set pull_required if the desired image is not yet loaded + set_fact: + pull_required: >- + {%- if image_reponame | regex_replace('^docker\.io/(library/)?', '') in docker_images.stdout.split(',') %}false{%- else -%}true{%- endif -%} + when: not download_always_pull + +- name: Check_pull_required | Check that the local digest sha256 corresponds to the given image tag + assert: + that: "{{ download.repo }}:{{ download.tag }} in docker_images.stdout.split(',')" + when: + - not download_always_pull + - not pull_required + - pull_by_digest + tags: + - asserts diff --git a/kubespray/roles/download/tasks/download_container.yml b/kubespray/roles/download/tasks/download_container.yml new file mode 100644 index 0000000..f98adfa --- /dev/null +++ b/kubespray/roles/download/tasks/download_container.yml @@ -0,0 +1,125 @@ +--- +- tags: + - download + block: + - name: Set default values for flag variables + set_fact: + image_is_cached: false + image_changed: false + pull_required: "{{ download_always_pull }}" + tags: + - facts + + - name: Download_container | Set a few facts + import_tasks: set_container_facts.yml + tags: + - facts + + - name: Download_container | Prepare container download + include_tasks: check_pull_required.yml + when: + - not download_always_pull + + - debug: # noqa name[missing] + msg: "Pull {{ image_reponame }} required is: {{ pull_required }}" + + - name: Download_container | Determine if image is in cache + stat: + path: "{{ image_path_cached }}" + get_attributes: no + get_checksum: no + get_mime: no + delegate_to: localhost + connection: local + delegate_facts: no + register: cache_image + changed_when: false + become: false + when: + - download_force_cache + + - name: Download_container | Set fact indicating if image is in cache + set_fact: + image_is_cached: "{{ cache_image.stat.exists }}" + tags: + - facts + when: + - download_force_cache + + - name: Stop if image not in cache on ansible host when download_force_cache=true + assert: + that: image_is_cached + msg: "Image cache file {{ image_path_cached }} not found for {{ image_reponame }} on localhost" + when: + - download_force_cache + - not download_run_once + + - name: Download_container | Download image if required + command: "{{ image_pull_command_on_localhost if download_localhost else image_pull_command }} {{ image_reponame }}" + delegate_to: "{{ download_delegate if download_run_once else inventory_hostname }}" + delegate_facts: yes + run_once: "{{ download_run_once }}" + register: pull_task_result + until: pull_task_result is succeeded + delay: "{{ retry_stagger | random + 3 }}" + retries: "{{ download_retries }}" + become: "{{ user_can_become_root | default(false) or not download_localhost }}" + environment: "{{ proxy_env if container_manager == 'containerd' else omit }}" + when: + - pull_required or download_run_once + - not image_is_cached + + - name: Download_container | Save and compress image + shell: "{{ image_save_command_on_localhost if download_localhost else image_save_command }}" # noqa command-instead-of-shell - image_save_command_on_localhost contains a pipe, therefore requires shell + delegate_to: "{{ download_delegate }}" + delegate_facts: no + register: container_save_status + failed_when: container_save_status.stderr + run_once: true + become: "{{ user_can_become_root | default(false) or not download_localhost }}" + when: + - not image_is_cached + - download_run_once + + - name: Download_container | Copy image to ansible host cache + ansible.posix.synchronize: + src: "{{ image_path_final }}" + dest: "{{ image_path_cached }}" + use_ssh_args: true + mode: pull + when: + - not image_is_cached + - download_run_once + - not download_localhost + - download_delegate == inventory_hostname + + - name: Download_container | Upload image to node if it is cached + ansible.posix.synchronize: + src: "{{ image_path_cached }}" + dest: "{{ image_path_final }}" + use_ssh_args: true + mode: push + delegate_facts: no + register: upload_image + failed_when: not upload_image + until: upload_image is succeeded + retries: "{{ download_retries }}" + delay: "{{ retry_stagger | random + 3 }}" + when: + - pull_required + - download_force_cache + + - name: Download_container | Load image into the local container registry + shell: "{{ image_load_command }}" # noqa command-instead-of-shell - image_load_command uses pipes, therefore requires shell + register: container_load_status + failed_when: container_load_status is failed + when: + - pull_required + - download_force_cache + + - name: Download_container | Remove container image from cache + file: + state: absent + path: "{{ image_path_final }}" + when: + - not download_keep_remote_cache diff --git a/kubespray/roles/download/tasks/download_file.yml b/kubespray/roles/download/tasks/download_file.yml new file mode 100644 index 0000000..f7f3080 --- /dev/null +++ b/kubespray/roles/download/tasks/download_file.yml @@ -0,0 +1,145 @@ +--- +- name: "Download_file | download {{ download.dest }}" + tags: + - download + block: + - name: Prep_download | Set a few facts + set_fact: + download_force_cache: "{{ true if download_run_once else download_force_cache }}" + + - name: Download_file | Starting download of file + debug: + msg: "{{ download.url }}" + run_once: "{{ download_run_once }}" + + - name: Download_file | Set pathname of cached file + set_fact: + file_path_cached: "{{ download_cache_dir }}/{{ download.dest | basename }}" + tags: + - facts + + - name: Download_file | Create dest directory on node + file: + path: "{{ download.dest | dirname }}" + owner: "{{ download.owner | default(omit) }}" + mode: 0755 + state: directory + recurse: yes + + - name: Download_file | Create local cache directory + file: + path: "{{ file_path_cached | dirname }}" + state: directory + recurse: yes + delegate_to: localhost + connection: local + delegate_facts: false + run_once: true + become: false + when: + - download_force_cache + tags: + - localhost + + - name: Download_file | Create cache directory on download_delegate host + file: + path: "{{ file_path_cached | dirname }}" + state: directory + recurse: yes + delegate_to: "{{ download_delegate }}" + delegate_facts: false + run_once: true + when: + - download_force_cache + - not download_localhost + + # We check a number of mirrors that may hold the file and pick a working one at random + # This task will avoid logging it's parameters to not leak environment passwords in the log + - name: Download_file | Validate mirrors + uri: + url: "{{ mirror }}" + method: HEAD + validate_certs: "{{ download_validate_certs }}" + url_username: "{{ download.username | default(omit) }}" + url_password: "{{ download.password | default(omit) }}" + force_basic_auth: "{{ download.force_basic_auth | default(omit) }}" + delegate_to: "{{ download_delegate if download_force_cache else inventory_hostname }}" + run_once: "{{ download_force_cache }}" + register: uri_result + become: "{{ not download_localhost }}" + until: uri_result is success + retries: "{{ download_retries }}" + delay: "{{ retry_stagger | default(5) }}" + environment: "{{ proxy_env }}" + no_log: "{{ not (unsafe_show_logs | bool) }}" + loop: "{{ download.mirrors | default([download.url]) }}" + loop_control: + loop_var: mirror + ignore_errors: true + + # Ansible 2.9 requires we convert a generator to a list + - name: Download_file | Get the list of working mirrors + set_fact: + valid_mirror_urls: "{{ uri_result.results | selectattr('failed', 'eq', False) | map(attribute='mirror') | list }}" + delegate_to: "{{ download_delegate if download_force_cache else inventory_hostname }}" + + # This must always be called, to check if the checksum matches. On no-match the file is re-downloaded. + # This task will avoid logging it's parameters to not leak environment passwords in the log + - name: Download_file | Download item + get_url: + url: "{{ valid_mirror_urls | random }}" + dest: "{{ file_path_cached if download_force_cache else download.dest }}" + owner: "{{ omit if download_localhost else (download.owner | default(omit)) }}" + mode: "{{ omit if download_localhost else (download.mode | default(omit)) }}" + checksum: "{{ 'sha256:' + download.sha256 if download.sha256 else omit }}" + validate_certs: "{{ download_validate_certs }}" + url_username: "{{ download.username | default(omit) }}" + url_password: "{{ download.password | default(omit) }}" + force_basic_auth: "{{ download.force_basic_auth | default(omit) }}" + timeout: "{{ download.timeout | default(omit) }}" + delegate_to: "{{ download_delegate if download_force_cache else inventory_hostname }}" + run_once: "{{ download_force_cache }}" + register: get_url_result + become: "{{ not download_localhost }}" + until: "'OK' in get_url_result.msg or + 'file already exists' in get_url_result.msg or + get_url_result.status_code == 304" + retries: "{{ download_retries }}" + delay: "{{ retry_stagger | default(5) }}" + environment: "{{ proxy_env }}" + no_log: "{{ not (unsafe_show_logs | bool) }}" + + - name: Download_file | Copy file back to ansible host file cache + ansible.posix.synchronize: + src: "{{ file_path_cached }}" + dest: "{{ file_path_cached }}" + use_ssh_args: true + mode: pull + when: + - download_force_cache + - not download_localhost + - download_delegate == inventory_hostname + + - name: Download_file | Copy file from cache to nodes, if it is available + ansible.posix.synchronize: + src: "{{ file_path_cached }}" + dest: "{{ download.dest }}" + use_ssh_args: true + mode: push + register: get_task + until: get_task is succeeded + delay: "{{ retry_stagger | random + 3 }}" + retries: "{{ download_retries }}" + when: + - download_force_cache + + - name: Download_file | Set mode and owner + file: + path: "{{ download.dest }}" + mode: "{{ download.mode | default(omit) }}" + owner: "{{ download.owner | default(omit) }}" + when: + - download_force_cache + + - name: "Download_file | Extract file archives" + include_tasks: "extract_file.yml" diff --git a/kubespray/roles/download/tasks/extract_file.yml b/kubespray/roles/download/tasks/extract_file.yml new file mode 100644 index 0000000..59d0531 --- /dev/null +++ b/kubespray/roles/download/tasks/extract_file.yml @@ -0,0 +1,11 @@ +--- +- name: Extract_file | Unpacking archive + unarchive: + src: "{{ download.dest }}" + dest: "{{ download.dest | dirname }}" + owner: "{{ download.owner | default(omit) }}" + mode: "{{ download.mode | default(omit) }}" + copy: no + extra_opts: "{{ download.unarchive_extra_opts | default(omit) }}" + when: + - download.unarchive | default(false) diff --git a/kubespray/roles/download/tasks/main.yml b/kubespray/roles/download/tasks/main.yml new file mode 100644 index 0000000..3309ab8 --- /dev/null +++ b/kubespray/roles/download/tasks/main.yml @@ -0,0 +1,30 @@ +--- +- name: Download | Prepare working directories and variables + import_tasks: prep_download.yml + when: + - not skip_downloads | default(false) + tags: + - download + - upload + +- name: Download | Get kubeadm binary and list of required images + include_tasks: prep_kubeadm_images.yml + when: + - not skip_downloads | default(false) + - inventory_hostname in groups['kube_control_plane'] + tags: + - download + - upload + +- name: Download | Download files / images + include_tasks: "{{ include_file }}" + loop: "{{ downloads | combine(kubeadm_images) | dict2items }}" + vars: + download: "{{ download_defaults | combine(item.value) }}" + include_file: "download_{% if download.container %}container{% else %}file{% endif %}.yml" + when: + - not skip_downloads | default(false) + - download.enabled + - item.value.enabled + - (not (item.value.container | default(false))) or (item.value.container and download_container) + - (download_run_once and inventory_hostname == download_delegate) or (group_names | intersect(download.groups) | length) diff --git a/kubespray/roles/download/tasks/prep_download.yml b/kubespray/roles/download/tasks/prep_download.yml new file mode 100644 index 0000000..4c737e8 --- /dev/null +++ b/kubespray/roles/download/tasks/prep_download.yml @@ -0,0 +1,92 @@ +--- +- name: Prep_download | Set a few facts + set_fact: + download_force_cache: "{{ true if download_run_once else download_force_cache }}" + tags: + - facts + +- name: Prep_download | On localhost, check if passwordless root is possible + command: "true" + delegate_to: localhost + connection: local + run_once: true + register: test_become + changed_when: false + ignore_errors: true # noqa ignore-errors + become: true + when: + - download_localhost + tags: + - localhost + - asserts + +- name: Prep_download | On localhost, check if user has access to the container runtime without using sudo + shell: "{{ image_info_command_on_localhost }}" # noqa command-instead-of-shell - image_info_command_on_localhost contains pipe, therefore requires shell + delegate_to: localhost + connection: local + run_once: true + register: test_docker + changed_when: false + ignore_errors: true # noqa ignore-errors + become: false + when: + - download_localhost + tags: + - localhost + - asserts + +- name: Prep_download | Parse the outputs of the previous commands + set_fact: + user_in_docker_group: "{{ not test_docker.failed }}" + user_can_become_root: "{{ not test_become.failed }}" + when: + - download_localhost + tags: + - localhost + - asserts + +- name: Prep_download | Check that local user is in group or can become root + assert: + that: "user_in_docker_group or user_can_become_root" + msg: >- + Error: User is not in docker group and cannot become root. When download_localhost is true, at least one of these two conditions must be met. + when: + - download_localhost + tags: + - localhost + - asserts + +- name: Prep_download | Register docker images info + shell: "{{ image_info_command }}" # noqa command-instead-of-shell - image_info_command contains pipe therefore requires shell + no_log: "{{ not (unsafe_show_logs | bool) }}" + register: docker_images + failed_when: false + changed_when: false + check_mode: no + when: download_container + +- name: Prep_download | Create staging directory on remote node + file: + path: "{{ local_release_dir }}/images" + state: directory + recurse: yes + mode: 0755 + owner: "{{ ansible_ssh_user | default(ansible_user_id) }}" + when: + - ansible_os_family not in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + +- name: Prep_download | Create local cache for files and images on control node + file: + path: "{{ download_cache_dir }}/images" + state: directory + recurse: yes + mode: 0755 + delegate_to: localhost + connection: local + delegate_facts: no + run_once: true + become: false + when: + - download_force_cache + tags: + - localhost diff --git a/kubespray/roles/download/tasks/prep_kubeadm_images.yml b/kubespray/roles/download/tasks/prep_kubeadm_images.yml new file mode 100644 index 0000000..fdfed1d --- /dev/null +++ b/kubespray/roles/download/tasks/prep_kubeadm_images.yml @@ -0,0 +1,71 @@ +--- +- name: Prep_kubeadm_images | Check kubeadm version matches kubernetes version + fail: + msg: "Kubeadm version {{ kubeadm_version }} do not matches kubernetes {{ kube_version }}" + when: + - not skip_downloads | default(false) + - not kubeadm_version == downloads.kubeadm.version + +- name: Prep_kubeadm_images | Download kubeadm binary + include_tasks: "download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.kubeadm) }}" + when: + - not skip_downloads | default(false) + - downloads.kubeadm.enabled + +- name: Prep_kubeadm_images | Create kubeadm config + template: + src: "kubeadm-images.yaml.j2" + dest: "{{ kube_config_dir }}/kubeadm-images.yaml" + mode: 0644 + when: + - not skip_kubeadm_images | default(false) + +- name: Prep_kubeadm_images | Copy kubeadm binary from download dir to system path + copy: + src: "{{ downloads.kubeadm.dest }}" + dest: "{{ bin_dir }}/kubeadm" + mode: 0755 + remote_src: true + +- name: Prep_kubeadm_images | Set kubeadm binary permissions + file: + path: "{{ bin_dir }}/kubeadm" + mode: "0755" + state: file + +- name: Prep_kubeadm_images | Generate list of required images + shell: "set -o pipefail && {{ bin_dir }}/kubeadm config images list --config={{ kube_config_dir }}/kubeadm-images.yaml | grep -Ev 'coredns|pause'" + args: + executable: /bin/bash + register: kubeadm_images_raw + run_once: true + changed_when: false + when: + - not skip_kubeadm_images | default(false) + +- name: Prep_kubeadm_images | Parse list of images + vars: + kubeadm_images_list: "{{ kubeadm_images_raw.stdout_lines }}" + set_fact: + kubeadm_image: + key: "kubeadm_{{ (item | regex_replace('^(?:.*\\/)*', '')).split(':')[0] }}" + value: + enabled: true + container: true + repo: "{{ item | regex_replace('^(.*):.*$', '\\1') }}" + tag: "{{ item | regex_replace('^.*:(.*)$', '\\1') }}" + groups: k8s_cluster + loop: "{{ kubeadm_images_list | flatten(levels=1) }}" + register: kubeadm_images_cooked + run_once: true + when: + - not skip_kubeadm_images | default(false) + +- name: Prep_kubeadm_images | Convert list of images to dict for later use + set_fact: + kubeadm_images: "{{ kubeadm_images_cooked.results | map(attribute='ansible_facts.kubeadm_image') | list | items2dict }}" + run_once: true + when: + - not skip_kubeadm_images | default(false) diff --git a/kubespray/roles/download/tasks/set_container_facts.yml b/kubespray/roles/download/tasks/set_container_facts.yml new file mode 100644 index 0000000..5b93f29 --- /dev/null +++ b/kubespray/roles/download/tasks/set_container_facts.yml @@ -0,0 +1,55 @@ +--- +- name: Set_container_facts | Display the name of the image being processed + debug: + msg: "{{ download.repo }}" + +- name: Set_container_facts | Set if containers should be pulled by digest + set_fact: + pull_by_digest: "{{ download.sha256 is defined and download.sha256 }}" + +- name: Set_container_facts | Define by what name to pull the image + set_fact: + image_reponame: >- + {%- if pull_by_digest %}{{ download.repo }}@sha256:{{ download.sha256 }}{%- else -%}{{ download.repo }}:{{ download.tag }}{%- endif -%} + +- name: Set_container_facts | Define file name of image + set_fact: + image_filename: "{{ image_reponame | regex_replace('/|\0|:', '_') }}.tar" + +- name: Set_container_facts | Define path of image + set_fact: + image_path_cached: "{{ download_cache_dir }}/images/{{ image_filename }}" + image_path_final: "{{ local_release_dir }}/images/{{ image_filename }}" + +- name: Set image save/load command for docker + set_fact: + image_save_command: "{{ docker_bin_dir }}/docker save {{ image_reponame }} | gzip -{{ download_compress }} > {{ image_path_final }}" + image_load_command: "{{ docker_bin_dir }}/docker load < {{ image_path_final }}" + when: container_manager == 'docker' + +- name: Set image save/load command for containerd + set_fact: + image_save_command: "{{ bin_dir }}/nerdctl -n k8s.io image save -o {{ image_path_final }} {{ image_reponame }}" + image_load_command: "{{ bin_dir }}/nerdctl -n k8s.io image load < {{ image_path_final }}" + when: container_manager == 'containerd' + +- name: Set image save/load command for crio + set_fact: + image_save_command: "{{ bin_dir }}/skopeo copy containers-storage:{{ image_reponame }} docker-archive:{{ image_path_final }} 2>/dev/null" + image_load_command: "{{ bin_dir }}/skopeo copy docker-archive:{{ image_path_final }} containers-storage:{{ image_reponame }} 2>/dev/null" + when: container_manager == 'crio' + +- name: Set image save/load command for docker on localhost + set_fact: + image_save_command_on_localhost: "{{ docker_bin_dir }}/docker save {{ image_reponame }} | gzip -{{ download_compress }} > {{ image_path_cached }}" + when: container_manager_on_localhost == 'docker' + +- name: Set image save/load command for containerd on localhost + set_fact: + image_save_command_on_localhost: "{{ containerd_bin_dir }}/ctr -n k8s.io image export --platform linux/{{ image_arch }} {{ image_path_cached }} {{ image_reponame }}" + when: container_manager_on_localhost == 'containerd' + +- name: Set image save/load command for crio on localhost + set_fact: + image_save_command_on_localhost: "{{ bin_dir }}/skopeo copy containers-storage:{{ image_reponame }} docker-archive:{{ image_path_final }} 2>/dev/null" + when: container_manager_on_localhost == 'crio' diff --git a/kubespray/roles/download/templates/kubeadm-images.yaml.j2 b/kubespray/roles/download/templates/kubeadm-images.yaml.j2 new file mode 100644 index 0000000..36154b3 --- /dev/null +++ b/kubespray/roles/download/templates/kubeadm-images.yaml.j2 @@ -0,0 +1,25 @@ +apiVersion: kubeadm.k8s.io/v1beta3 +kind: InitConfiguration +nodeRegistration: + criSocket: {{ cri_socket }} +--- +apiVersion: kubeadm.k8s.io/v1beta3 +kind: ClusterConfiguration +imageRepository: {{ kube_image_repo }} +kubernetesVersion: {{ kube_version }} +etcd: +{% if etcd_deployment_type == "kubeadm" %} + local: + imageRepository: "{{ etcd_image_repo | regex_replace("/etcd$","") }}" + imageTag: "{{ etcd_image_tag }}" +{% else %} + external: + endpoints: +{% for endpoint in etcd_access_addresses.split(',') %} + - {{ endpoint }} +{% endfor %} +{% endif %} +dns: + type: CoreDNS + imageRepository: {{ coredns_image_repo | regex_replace('/coredns(?!/coredns).*$', '') }} + imageTag: {{ coredns_image_tag }} diff --git a/kubespray/roles/etcd/defaults/main.yml b/kubespray/roles/etcd/defaults/main.yml new file mode 100644 index 0000000..bf38ace --- /dev/null +++ b/kubespray/roles/etcd/defaults/main.yml @@ -0,0 +1,122 @@ +--- +# Set etcd user +etcd_owner: etcd + +# Set to false to only do certificate management +etcd_cluster_setup: true +etcd_events_cluster_setup: false + +# Set to true to separate k8s events to a different etcd cluster +etcd_events_cluster_enabled: false + +etcd_backup_prefix: "/var/backups" +etcd_data_dir: "/var/lib/etcd" + +# Number of etcd backups to retain. Set to a value < 0 to retain all backups +etcd_backup_retention_count: -1 + +force_etcd_cert_refresh: true +etcd_config_dir: /etc/ssl/etcd +etcd_cert_dir: "{{ etcd_config_dir }}/ssl" +etcd_cert_dir_mode: "0700" +etcd_cert_group: root +# Note: This does not set up DNS entries. It simply adds the following DNS +# entries to the certificate +etcd_cert_alt_names: + - "etcd.kube-system.svc.{{ dns_domain }}" + - "etcd.kube-system.svc" + - "etcd.kube-system" + - "etcd" +etcd_cert_alt_ips: [] + +etcd_script_dir: "{{ bin_dir }}/etcd-scripts" + +etcd_heartbeat_interval: "250" +etcd_election_timeout: "5000" + +# etcd_snapshot_count: "10000" + +etcd_metrics: "basic" + +# Define in inventory to set a separate port for etcd to expose metrics on +# etcd_metrics_port: 2381 + +## A dictionary of extra environment variables to add to etcd.env, formatted like: +## etcd_extra_vars: +## ETCD_VAR1: "value1" +## ETCD_VAR2: "value2" +etcd_extra_vars: {} + +# Limits +# Limit memory only if <4GB memory on host. 0=unlimited +# This value is only relevant when deploying etcd with `etcd_deployment_type: docker` +etcd_memory_limit: "{% if ansible_memtotal_mb < 4096 %}512M{% else %}0{% endif %}" + +# The default storage size limit is 2G. +# 8G is a suggested maximum size for normal environments and etcd warns at startup if the configured value exceeds it. +# etcd_quota_backend_bytes: "2147483648" + +# Maximum client request size in bytes the server will accept. +# etcd is designed to handle small key value pairs typical for metadata. +# Larger requests will work, but may increase the latency of other requests +# etcd_max_request_bytes: "1572864" + +# Uncomment to set CPU share for etcd +# etcd_cpu_limit: 300m + +etcd_blkio_weight: 1000 + +etcd_node_cert_hosts: "{{ groups['k8s_cluster'] }}" + +etcd_compaction_retention: "8" + +# Force clients like etcdctl to use TLS certs (different than peer security) +etcd_secure_client: true + +# Enable peer client cert authentication +etcd_peer_client_auth: true + +# Maximum number of snapshot files to retain (0 is unlimited) +# etcd_max_snapshots: 5 + +# Maximum number of wal files to retain (0 is unlimited) +# etcd_max_wals: 5 + +# Number of loop retries +etcd_retries: 4 + +## Support tls cipher suites. +# etcd_tls_cipher_suites: {} +# - TLS_RSA_WITH_RC4_128_SHA +# - TLS_RSA_WITH_3DES_EDE_CBC_SHA +# - TLS_RSA_WITH_AES_128_CBC_SHA +# - TLS_RSA_WITH_AES_256_CBC_SHA +# - TLS_RSA_WITH_AES_128_CBC_SHA256 +# - TLS_RSA_WITH_AES_128_GCM_SHA256 +# - TLS_RSA_WITH_AES_256_GCM_SHA384 +# - TLS_ECDHE_ECDSA_WITH_RC4_128_SHA +# - TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA +# - TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA +# - TLS_ECDHE_RSA_WITH_RC4_128_SHA +# - TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA +# - TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256 +# - TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256 +# - TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 +# - TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 +# - TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 +# - TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 +# - TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305 +# - TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256 +# - TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305 +# - TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256 + +# ETCD 3.5.x issue +# https://groups.google.com/a/kubernetes.io/g/dev/c/B7gJs88XtQc/m/rSgNOzV2BwAJ?utm_medium=email&utm_source=footer +etcd_experimental_initial_corrupt_check: true + +# If this is true, debug information will be displayed but +# may contain some private data, so it is recommended to set it to false +# in the production environment. +unsafe_show_logs: false diff --git a/kubespray/roles/etcd/handlers/backup.yml b/kubespray/roles/etcd/handlers/backup.yml new file mode 100644 index 0000000..2c55778 --- /dev/null +++ b/kubespray/roles/etcd/handlers/backup.yml @@ -0,0 +1,63 @@ +--- +- name: Backup etcd data + command: /bin/true + notify: + - Refresh Time Fact + - Set Backup Directory + - Create Backup Directory + - Stat etcd v2 data directory + - Backup etcd v2 data + - Backup etcd v3 data + when: etcd_cluster_is_healthy.rc == 0 + +- name: Refresh Time Fact + setup: + filter: ansible_date_time + +- name: Set Backup Directory + set_fact: + etcd_backup_directory: "{{ etcd_backup_prefix }}/etcd-{{ ansible_date_time.date }}_{{ ansible_date_time.time }}" + +- name: Create Backup Directory + file: + path: "{{ etcd_backup_directory }}" + state: directory + owner: root + group: root + mode: 0600 + +- name: Stat etcd v2 data directory + stat: + path: "{{ etcd_data_dir }}/member" + get_attributes: no + get_checksum: no + get_mime: no + register: etcd_data_dir_member + +- name: Backup etcd v2 data + when: etcd_data_dir_member.stat.exists + command: >- + {{ bin_dir }}/etcdctl backup + --data-dir {{ etcd_data_dir }} + --backup-dir {{ etcd_backup_directory }} + environment: + ETCDCTL_API: "2" + retries: 3 + register: backup_v2_command + until: backup_v2_command.rc == 0 + delay: "{{ retry_stagger | random + 3 }}" + +- name: Backup etcd v3 data + command: >- + {{ bin_dir }}/etcdctl + snapshot save {{ etcd_backup_directory }}/snapshot.db + environment: + ETCDCTL_API: "3" + ETCDCTL_ENDPOINTS: "{{ etcd_access_addresses.split(',') | first }}" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + retries: 3 + register: etcd_backup_v3_command + until: etcd_backup_v3_command.rc == 0 + delay: "{{ retry_stagger | random + 3 }}" diff --git a/kubespray/roles/etcd/handlers/backup_cleanup.yml b/kubespray/roles/etcd/handlers/backup_cleanup.yml new file mode 100644 index 0000000..3cebfd0 --- /dev/null +++ b/kubespray/roles/etcd/handlers/backup_cleanup.yml @@ -0,0 +1,12 @@ +--- +- name: Cleanup etcd backups + command: /bin/true + notify: + - Remove old etcd backups + +- name: Remove old etcd backups + shell: + chdir: "{{ etcd_backup_prefix }}" + cmd: "set -o pipefail && find . -name 'etcd-*' -type d | sort -n | head -n -{{ etcd_backup_retention_count }} | xargs rm -rf" + executable: /bin/bash + when: etcd_backup_retention_count >= 0 diff --git a/kubespray/roles/etcd/handlers/main.yml b/kubespray/roles/etcd/handlers/main.yml new file mode 100644 index 0000000..f09789c --- /dev/null +++ b/kubespray/roles/etcd/handlers/main.yml @@ -0,0 +1,64 @@ +--- +- name: Restart etcd + command: /bin/true + notify: + - Backup etcd data + - Etcd | reload systemd + - Reload etcd + - Wait for etcd up + - Cleanup etcd backups + +- name: Restart etcd-events + command: /bin/true + notify: + - Etcd | reload systemd + - Reload etcd-events + - Wait for etcd-events up + +- name: Backup etcd + import_tasks: backup.yml + +- name: Etcd | reload systemd + systemd: + daemon_reload: true + +- name: Reload etcd + service: + name: etcd + state: restarted + when: is_etcd_master + +- name: Reload etcd-events + service: + name: etcd-events + state: restarted + when: is_etcd_master + +- name: Wait for etcd up + uri: + url: "https://{% if is_etcd_master %}{{ etcd_address }}{% else %}127.0.0.1{% endif %}:2379/health" + validate_certs: no + client_cert: "{{ etcd_cert_dir }}/member-{{ inventory_hostname }}.pem" + client_key: "{{ etcd_cert_dir }}/member-{{ inventory_hostname }}-key.pem" + register: result + until: result.status is defined and result.status == 200 + retries: 60 + delay: 1 + +- name: Cleanup etcd backups + import_tasks: backup_cleanup.yml + +- name: Wait for etcd-events up + uri: + url: "https://{% if is_etcd_master %}{{ etcd_address }}{% else %}127.0.0.1{% endif %}:2383/health" + validate_certs: no + client_cert: "{{ etcd_cert_dir }}/member-{{ inventory_hostname }}.pem" + client_key: "{{ etcd_cert_dir }}/member-{{ inventory_hostname }}-key.pem" + register: result + until: result.status is defined and result.status == 200 + retries: 60 + delay: 1 + +- name: Set etcd_secret_changed + set_fact: + etcd_secret_changed: true diff --git a/kubespray/roles/etcd/meta/main.yml b/kubespray/roles/etcd/meta/main.yml new file mode 100644 index 0000000..e996646 --- /dev/null +++ b/kubespray/roles/etcd/meta/main.yml @@ -0,0 +1,8 @@ +--- +dependencies: + - role: adduser + user: "{{ addusers.etcd }}" + when: not (ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk", "ClearLinux"] or is_fedora_coreos) + - role: adduser + user: "{{ addusers.kube }}" + when: not (ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk", "ClearLinux"] or is_fedora_coreos) diff --git a/kubespray/roles/etcd/tasks/check_certs.yml b/kubespray/roles/etcd/tasks/check_certs.yml new file mode 100644 index 0000000..2cb802d --- /dev/null +++ b/kubespray/roles/etcd/tasks/check_certs.yml @@ -0,0 +1,171 @@ +--- +- name: "Check_certs | Register certs that have already been generated on first etcd node" + find: + paths: "{{ etcd_cert_dir }}" + patterns: "ca.pem,node*.pem,member*.pem,admin*.pem" + get_checksum: true + delegate_to: "{{ groups['etcd'][0] }}" + register: etcdcert_master + run_once: true + +- name: "Check_certs | Set default value for 'sync_certs', 'gen_certs' and 'etcd_secret_changed' to false" + set_fact: + sync_certs: false + gen_certs: false + etcd_secret_changed: false + +- name: "Check certs | Register ca and etcd admin/member certs on etcd hosts" + stat: + path: "{{ etcd_cert_dir }}/{{ item }}" + get_attributes: no + get_checksum: yes + get_mime: no + register: etcd_member_certs + when: inventory_hostname in groups['etcd'] + with_items: + - ca.pem + - member-{{ inventory_hostname }}.pem + - member-{{ inventory_hostname }}-key.pem + - admin-{{ inventory_hostname }}.pem + - admin-{{ inventory_hostname }}-key.pem + +- name: "Check certs | Register ca and etcd node certs on kubernetes hosts" + stat: + path: "{{ etcd_cert_dir }}/{{ item }}" + register: etcd_node_certs + when: inventory_hostname in groups['k8s_cluster'] + with_items: + - ca.pem + - node-{{ inventory_hostname }}.pem + - node-{{ inventory_hostname }}-key.pem + +- name: "Check_certs | Set 'gen_certs' to true if expected certificates are not on the first etcd node(1/2)" + set_fact: + gen_certs: true + when: force_etcd_cert_refresh or not item in etcdcert_master.files | map(attribute='path') | list + run_once: true + with_items: "{{ expected_files }}" + vars: + expected_files: >- + ['{{ etcd_cert_dir }}/ca.pem', + {% set etcd_members = groups['etcd'] %} + {% for host in etcd_members %} + '{{ etcd_cert_dir }}/admin-{{ host }}.pem', + '{{ etcd_cert_dir }}/admin-{{ host }}-key.pem', + '{{ etcd_cert_dir }}/member-{{ host }}.pem', + '{{ etcd_cert_dir }}/member-{{ host }}-key.pem', + {% endfor %} + {% set k8s_nodes = groups['kube_control_plane'] %} + {% for host in k8s_nodes %} + '{{ etcd_cert_dir }}/node-{{ host }}.pem', + '{{ etcd_cert_dir }}/node-{{ host }}-key.pem' + {% if not loop.last %}{{ ',' }}{% endif %} + {% endfor %}] + +- name: "Check_certs | Set 'gen_certs' to true if expected certificates are not on the first etcd node(2/2)" + set_fact: + gen_certs: true + run_once: true + with_items: "{{ expected_files }}" + vars: + expected_files: >- + ['{{ etcd_cert_dir }}/ca.pem', + {% set etcd_members = groups['etcd'] %} + {% for host in etcd_members %} + '{{ etcd_cert_dir }}/admin-{{ host }}.pem', + '{{ etcd_cert_dir }}/admin-{{ host }}-key.pem', + '{{ etcd_cert_dir }}/member-{{ host }}.pem', + '{{ etcd_cert_dir }}/member-{{ host }}-key.pem', + {% endfor %} + {% set k8s_nodes = groups['k8s_cluster'] | unique | sort %} + {% for host in k8s_nodes %} + '{{ etcd_cert_dir }}/node-{{ host }}.pem', + '{{ etcd_cert_dir }}/node-{{ host }}-key.pem' + {% if not loop.last %}{{ ',' }}{% endif %} + {% endfor %}] + when: + - kube_network_plugin in ["calico", "flannel", "cilium"] or cilium_deploy_additionally | default(false) | bool + - kube_network_plugin != "calico" or calico_datastore == "etcd" + - force_etcd_cert_refresh or not item in etcdcert_master.files | map(attribute='path') | list + +- name: "Check_certs | Set 'gen_master_certs' object to track whether member and admin certs exist on first etcd node" + set_fact: + # noqa: jinja[spacing] + gen_master_certs: |- + { + {% set etcd_members = groups['etcd'] -%} + {% set existing_certs = etcdcert_master.files | map(attribute='path') | list | sort %} + {% for host in etcd_members -%} + {% set member_cert = "%s/member-%s.pem" | format(etcd_cert_dir, host) %} + {% set member_key = "%s/member-%s-key.pem" | format(etcd_cert_dir, host) %} + {% set admin_cert = "%s/admin-%s.pem" | format(etcd_cert_dir, host) %} + {% set admin_key = "%s/admin-%s-key.pem" | format(etcd_cert_dir, host) %} + {% if force_etcd_cert_refresh -%} + "{{ host }}": True, + {% elif member_cert in existing_certs and member_key in existing_certs and admin_cert in existing_certs and admin_key in existing_certs -%} + "{{ host }}": False, + {% else -%} + "{{ host }}": True, + {% endif -%} + {% endfor %} + } + run_once: true + +- name: "Check_certs | Set 'gen_node_certs' object to track whether node certs exist on first etcd node" + set_fact: + # noqa: jinja[spacing] + gen_node_certs: |- + { + {% set k8s_nodes = groups['k8s_cluster'] -%} + {% set existing_certs = etcdcert_master.files | map(attribute='path') | list | sort %} + {% for host in k8s_nodes -%} + {% set host_cert = "%s/node-%s.pem" | format(etcd_cert_dir, host) %} + {% set host_key = "%s/node-%s-key.pem" | format(etcd_cert_dir, host) %} + {% if force_etcd_cert_refresh -%} + "{{ host }}": True, + {% elif host_cert in existing_certs and host_key in existing_certs -%} + "{{ host }}": False, + {% else -%} + "{{ host }}": True, + {% endif -%} + {% endfor %} + } + run_once: true + +- name: "Check_certs | Set 'etcd_member_requires_sync' to true if ca or member/admin cert and key don't exist on etcd member or checksum doesn't match" + set_fact: + etcd_member_requires_sync: true + when: + - inventory_hostname in groups['etcd'] + - (not etcd_member_certs.results[0].stat.exists | default(false)) or + (not etcd_member_certs.results[1].stat.exists | default(false)) or + (not etcd_member_certs.results[2].stat.exists | default(false)) or + (not etcd_member_certs.results[3].stat.exists | default(false)) or + (not etcd_member_certs.results[4].stat.exists | default(false)) or + (etcd_member_certs.results[0].stat.checksum | default('') != etcdcert_master.files | selectattr("path", "equalto", etcd_member_certs.results[0].stat.path) | map(attribute="checksum") | first | default('')) or + (etcd_member_certs.results[1].stat.checksum | default('') != etcdcert_master.files | selectattr("path", "equalto", etcd_member_certs.results[1].stat.path) | map(attribute="checksum") | first | default('')) or + (etcd_member_certs.results[2].stat.checksum | default('') != etcdcert_master.files | selectattr("path", "equalto", etcd_member_certs.results[2].stat.path) | map(attribute="checksum") | first | default('')) or + (etcd_member_certs.results[3].stat.checksum | default('') != etcdcert_master.files | selectattr("path", "equalto", etcd_member_certs.results[3].stat.path) | map(attribute="checksum") | first | default('')) or + (etcd_member_certs.results[4].stat.checksum | default('') != etcdcert_master.files | selectattr("path", "equalto", etcd_member_certs.results[4].stat.path) | map(attribute="checksum") | first | default('')) + +- name: "Check_certs | Set 'kubernetes_host_requires_sync' to true if ca or node cert and key don't exist on kubernetes host or checksum doesn't match" + set_fact: + kubernetes_host_requires_sync: true + when: + - inventory_hostname in groups['k8s_cluster'] and + inventory_hostname not in groups['etcd'] + - (not etcd_node_certs.results[0].stat.exists | default(false)) or + (not etcd_node_certs.results[1].stat.exists | default(false)) or + (not etcd_node_certs.results[2].stat.exists | default(false)) or + (etcd_node_certs.results[0].stat.checksum | default('') != etcdcert_master.files | selectattr("path", "equalto", etcd_node_certs.results[0].stat.path) | map(attribute="checksum") | first | default('')) or + (etcd_node_certs.results[1].stat.checksum | default('') != etcdcert_master.files | selectattr("path", "equalto", etcd_node_certs.results[1].stat.path) | map(attribute="checksum") | first | default('')) or + (etcd_node_certs.results[2].stat.checksum | default('') != etcdcert_master.files | selectattr("path", "equalto", etcd_node_certs.results[2].stat.path) | map(attribute="checksum") | first | default('')) + +- name: "Check_certs | Set 'sync_certs' to true" + set_fact: + sync_certs: true + when: + - etcd_member_requires_sync | default(false) or + kubernetes_host_requires_sync | default(false) or + (inventory_hostname in gen_master_certs and gen_master_certs[inventory_hostname]) or + (inventory_hostname in gen_node_certs and gen_node_certs[inventory_hostname]) diff --git a/kubespray/roles/etcd/tasks/configure.yml b/kubespray/roles/etcd/tasks/configure.yml new file mode 100644 index 0000000..f1d6a48 --- /dev/null +++ b/kubespray/roles/etcd/tasks/configure.yml @@ -0,0 +1,173 @@ +--- +- name: Configure | Check if etcd cluster is healthy + shell: "set -o pipefail && {{ bin_dir }}/etcdctl endpoint --cluster status && {{ bin_dir }}/etcdctl endpoint --cluster health 2>&1 | grep -v 'Error: unhealthy cluster' >/dev/null" + args: + executable: /bin/bash + register: etcd_cluster_is_healthy + failed_when: false + changed_when: false + check_mode: no + run_once: yes + when: + - is_etcd_master + - etcd_cluster_setup + tags: + - facts + environment: + ETCDCTL_API: "3" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + ETCDCTL_ENDPOINTS: "{{ etcd_access_addresses }}" + +- name: Configure | Check if etcd-events cluster is healthy + shell: "set -o pipefail && {{ bin_dir }}/etcdctl endpoint --cluster status && {{ bin_dir }}/etcdctl endpoint --cluster health 2>&1 | grep -v 'Error: unhealthy cluster' >/dev/null" + args: + executable: /bin/bash + register: etcd_events_cluster_is_healthy + failed_when: false + changed_when: false + check_mode: no + run_once: yes + when: + - is_etcd_master + - etcd_events_cluster_setup + tags: + - facts + environment: + ETCDCTL_API: "3" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + ETCDCTL_ENDPOINTS: "{{ etcd_events_access_addresses }}" + +- name: Configure | Refresh etcd config + include_tasks: refresh_config.yml + when: is_etcd_master + +- name: Configure | Copy etcd.service systemd file + template: + src: "etcd-{{ etcd_deployment_type }}.service.j2" + dest: /etc/systemd/system/etcd.service + backup: yes + mode: 0644 + when: is_etcd_master and etcd_cluster_setup + +- name: Configure | Copy etcd-events.service systemd file + template: + src: "etcd-events-{{ etcd_deployment_type }}.service.j2" + dest: /etc/systemd/system/etcd-events.service + backup: yes + mode: 0644 + when: is_etcd_master and etcd_events_cluster_setup + +- name: Configure | reload systemd + systemd: + daemon_reload: true + when: is_etcd_master + +# when scaling new etcd will fail to start +- name: Configure | Ensure etcd is running + service: + name: etcd + state: started + enabled: yes + ignore_errors: "{{ etcd_cluster_is_healthy.rc == 0 }}" # noqa ignore-errors + when: is_etcd_master and etcd_cluster_setup + +# when scaling new etcd will fail to start +- name: Configure | Ensure etcd-events is running + service: + name: etcd-events + state: started + enabled: yes + ignore_errors: "{{ etcd_events_cluster_is_healthy.rc != 0 }}" # noqa ignore-errors + when: is_etcd_master and etcd_events_cluster_setup + +- name: Configure | Wait for etcd cluster to be healthy + shell: "set -o pipefail && {{ bin_dir }}/etcdctl endpoint --cluster status && {{ bin_dir }}/etcdctl endpoint --cluster health 2>&1 | grep -v 'Error: unhealthy cluster' >/dev/null" + args: + executable: /bin/bash + register: etcd_cluster_is_healthy + until: etcd_cluster_is_healthy.rc == 0 + retries: "{{ etcd_retries }}" + delay: "{{ retry_stagger | random + 3 }}" + changed_when: false + check_mode: no + run_once: yes + when: + - is_etcd_master + - etcd_cluster_setup + tags: + - facts + environment: + ETCDCTL_API: "3" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + ETCDCTL_ENDPOINTS: "{{ etcd_access_addresses }}" + +- name: Configure | Wait for etcd-events cluster to be healthy + shell: "set -o pipefail && {{ bin_dir }}/etcdctl endpoint --cluster status && {{ bin_dir }}/etcdctl endpoint --cluster health 2>&1 | grep -v 'Error: unhealthy cluster' >/dev/null" + args: + executable: /bin/bash + register: etcd_events_cluster_is_healthy + until: etcd_events_cluster_is_healthy.rc == 0 + retries: "{{ etcd_retries }}" + delay: "{{ retry_stagger | random + 3 }}" + changed_when: false + check_mode: no + run_once: yes + when: + - is_etcd_master + - etcd_events_cluster_setup + tags: + - facts + environment: + ETCDCTL_API: "3" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + ETCDCTL_ENDPOINTS: "{{ etcd_events_access_addresses }}" + +- name: Configure | Check if member is in etcd cluster + shell: "{{ bin_dir }}/etcdctl member list | grep -w -q {{ etcd_access_address }}" + register: etcd_member_in_cluster + ignore_errors: true # noqa ignore-errors + changed_when: false + check_mode: no + when: is_etcd_master and etcd_cluster_setup + tags: + - facts + environment: + ETCDCTL_API: "3" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + ETCDCTL_ENDPOINTS: "{{ etcd_access_addresses }}" + +- name: Configure | Check if member is in etcd-events cluster + shell: "{{ bin_dir }}/etcdctl member list | grep -w -q {{ etcd_access_address }}" + register: etcd_events_member_in_cluster + ignore_errors: true # noqa ignore-errors + changed_when: false + check_mode: no + when: is_etcd_master and etcd_events_cluster_setup + tags: + - facts + environment: + ETCDCTL_API: "3" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + ETCDCTL_ENDPOINTS: "{{ etcd_events_access_addresses }}" + +- name: Configure | Join member(s) to etcd cluster one at a time + include_tasks: join_etcd_member.yml + with_items: "{{ groups['etcd'] }}" + when: inventory_hostname == item and etcd_cluster_setup and etcd_member_in_cluster.rc != 0 and etcd_cluster_is_healthy.rc == 0 + +- name: Configure | Join member(s) to etcd-events cluster one at a time + include_tasks: join_etcd-events_member.yml + with_items: "{{ groups['etcd'] }}" + when: inventory_hostname == item and etcd_events_cluster_setup and etcd_events_member_in_cluster.rc != 0 and etcd_events_cluster_is_healthy.rc == 0 diff --git a/kubespray/roles/etcd/tasks/gen_certs_script.yml b/kubespray/roles/etcd/tasks/gen_certs_script.yml new file mode 100644 index 0000000..2ce3e14 --- /dev/null +++ b/kubespray/roles/etcd/tasks/gen_certs_script.yml @@ -0,0 +1,171 @@ +--- +- name: Gen_certs | create etcd cert dir + file: + path: "{{ etcd_cert_dir }}" + group: "{{ etcd_cert_group }}" + state: directory + owner: "{{ etcd_owner }}" + mode: "{{ etcd_cert_dir_mode }}" + recurse: yes + +- name: "Gen_certs | create etcd script dir (on {{ groups['etcd'][0] }})" + file: + path: "{{ etcd_script_dir }}" + state: directory + owner: root + mode: 0700 + run_once: yes + when: inventory_hostname == groups['etcd'][0] + +- name: Gen_certs | write openssl config + template: + src: "openssl.conf.j2" + dest: "{{ etcd_config_dir }}/openssl.conf" + mode: 0640 + run_once: yes + delegate_to: "{{ groups['etcd'][0] }}" + when: + - gen_certs | default(false) + - inventory_hostname == groups['etcd'][0] + +- name: Gen_certs | copy certs generation script + template: + src: "make-ssl-etcd.sh.j2" + dest: "{{ etcd_script_dir }}/make-ssl-etcd.sh" + mode: 0700 + run_once: yes + when: + - gen_certs | default(false) + - inventory_hostname == groups['etcd'][0] + +- name: Gen_certs | run cert generation script for etcd and kube control plane nodes + command: "bash -x {{ etcd_script_dir }}/make-ssl-etcd.sh -f {{ etcd_config_dir }}/openssl.conf -d {{ etcd_cert_dir }}" + environment: + MASTERS: |- + {% for m in groups['etcd'] %} + {% if gen_master_certs[m] %} + {{ m }} + {% endif %} + {% endfor %} + HOSTS: |- + {% for h in groups['kube_control_plane'] %} + {% if gen_node_certs[h] %} + {{ h }} + {% endif %} + {% endfor %} + run_once: yes + delegate_to: "{{ groups['etcd'][0] }}" + when: gen_certs | default(false) + notify: Set etcd_secret_changed + +- name: Gen_certs | run cert generation script for all clients + command: "bash -x {{ etcd_script_dir }}/make-ssl-etcd.sh -f {{ etcd_config_dir }}/openssl.conf -d {{ etcd_cert_dir }}" + environment: + HOSTS: |- + {% for h in groups['k8s_cluster'] %} + {% if gen_node_certs[h] %} + {{ h }} + {% endif %} + {% endfor %} + run_once: yes + delegate_to: "{{ groups['etcd'][0] }}" + when: + - kube_network_plugin in ["calico", "flannel", "cilium"] or cilium_deploy_additionally | default(false) | bool + - kube_network_plugin != "calico" or calico_datastore == "etcd" + - gen_certs | default(false) + notify: Set etcd_secret_changed + +- name: Gen_certs | Gather etcd member/admin and kube_control_plane client certs from first etcd node + slurp: + src: "{{ item }}" + register: etcd_master_certs + with_items: + - "{{ etcd_cert_dir }}/ca.pem" + - "{{ etcd_cert_dir }}/ca-key.pem" + - "[{% for node in groups['etcd'] %} + '{{ etcd_cert_dir }}/admin-{{ node }}.pem', + '{{ etcd_cert_dir }}/admin-{{ node }}-key.pem', + '{{ etcd_cert_dir }}/member-{{ node }}.pem', + '{{ etcd_cert_dir }}/member-{{ node }}-key.pem', + {% endfor %}]" + - "[{% for node in (groups['kube_control_plane']) %} + '{{ etcd_cert_dir }}/node-{{ node }}.pem', + '{{ etcd_cert_dir }}/node-{{ node }}-key.pem', + {% endfor %}]" + delegate_to: "{{ groups['etcd'][0] }}" + when: + - inventory_hostname in groups['etcd'] + - sync_certs | default(false) + - inventory_hostname != groups['etcd'][0] + notify: Set etcd_secret_changed + +- name: Gen_certs | Write etcd member/admin and kube_control_plane client certs to other etcd nodes + copy: + dest: "{{ item.item }}" + content: "{{ item.content | b64decode }}" + group: "{{ etcd_cert_group }}" + owner: "{{ etcd_owner }}" + mode: 0640 + with_items: "{{ etcd_master_certs.results }}" + when: + - inventory_hostname in groups['etcd'] + - sync_certs | default(false) + - inventory_hostname != groups['etcd'][0] + loop_control: + label: "{{ item.item }}" + +- name: Gen_certs | Gather node certs from first etcd node + slurp: + src: "{{ item }}" + register: etcd_master_node_certs + with_items: + - "[{% for node in groups['k8s_cluster'] %} + '{{ etcd_cert_dir }}/node-{{ node }}.pem', + '{{ etcd_cert_dir }}/node-{{ node }}-key.pem', + {% endfor %}]" + delegate_to: "{{ groups['etcd'][0] }}" + when: + - inventory_hostname in groups['etcd'] + - inventory_hostname != groups['etcd'][0] + - kube_network_plugin in ["calico", "flannel", "cilium"] or cilium_deploy_additionally | default(false) | bool + - kube_network_plugin != "calico" or calico_datastore == "etcd" + notify: Set etcd_secret_changed + +- name: Gen_certs | Write node certs to other etcd nodes + copy: + dest: "{{ item.item }}" + content: "{{ item.content | b64decode }}" + group: "{{ etcd_cert_group }}" + owner: "{{ etcd_owner }}" + mode: 0640 + with_items: "{{ etcd_master_node_certs.results }}" + when: + - inventory_hostname in groups['etcd'] + - inventory_hostname != groups['etcd'][0] + - kube_network_plugin in ["calico", "flannel", "cilium"] or cilium_deploy_additionally | default(false) | bool + - kube_network_plugin != "calico" or calico_datastore == "etcd" + loop_control: + label: "{{ item.item }}" + +- name: Gen_certs | Generate etcd certs + include_tasks: gen_nodes_certs_script.yml + when: + - inventory_hostname in groups['kube_control_plane'] and + sync_certs | default(false) and inventory_hostname not in groups['etcd'] + +- name: Gen_certs | Generate etcd certs on nodes if needed + include_tasks: gen_nodes_certs_script.yml + when: + - kube_network_plugin in ["calico", "flannel", "cilium"] or cilium_deploy_additionally | default(false) | bool + - kube_network_plugin != "calico" or calico_datastore == "etcd" + - inventory_hostname in groups['k8s_cluster'] and + sync_certs | default(false) and inventory_hostname not in groups['etcd'] + +- name: Gen_certs | check certificate permissions + file: + path: "{{ etcd_cert_dir }}" + group: "{{ etcd_cert_group }}" + state: directory + owner: "{{ etcd_owner }}" + mode: "{{ etcd_cert_dir_mode }}" + recurse: yes diff --git a/kubespray/roles/etcd/tasks/gen_nodes_certs_script.yml b/kubespray/roles/etcd/tasks/gen_nodes_certs_script.yml new file mode 100644 index 0000000..a7b31db --- /dev/null +++ b/kubespray/roles/etcd/tasks/gen_nodes_certs_script.yml @@ -0,0 +1,31 @@ +--- +- name: Gen_certs | Set cert names per node + set_fact: + my_etcd_node_certs: [ 'ca.pem', + 'node-{{ inventory_hostname }}.pem', + 'node-{{ inventory_hostname }}-key.pem'] + tags: + - facts + +- name: "Check_certs | Set 'sync_certs' to true on nodes" + set_fact: + sync_certs: true + with_items: + - "{{ my_etcd_node_certs }}" + +- name: Gen_certs | Gather node certs + shell: "set -o pipefail && tar cfz - -C {{ etcd_cert_dir }} {{ my_etcd_node_certs | join(' ') }} | base64 --wrap=0" + args: + executable: /bin/bash + no_log: "{{ not (unsafe_show_logs | bool) }}" + register: etcd_node_certs + check_mode: no + delegate_to: "{{ groups['etcd'][0] }}" + changed_when: false + +- name: Gen_certs | Copy certs on nodes + shell: "set -o pipefail && base64 -d <<< '{{ etcd_node_certs.stdout | quote }}' | tar xz -C {{ etcd_cert_dir }}" + args: + executable: /bin/bash + no_log: "{{ not (unsafe_show_logs | bool) }}" + changed_when: false diff --git a/kubespray/roles/etcd/tasks/install_docker.yml b/kubespray/roles/etcd/tasks/install_docker.yml new file mode 100644 index 0000000..cc2fdec --- /dev/null +++ b/kubespray/roles/etcd/tasks/install_docker.yml @@ -0,0 +1,42 @@ +--- +- name: Get currently-deployed etcd version + shell: "{{ docker_bin_dir }}/docker ps --filter='name={{ etcd_member_name }}' --format='{{ '{{ .Image }}' }}'" + register: etcd_current_docker_image + when: etcd_cluster_setup + +- name: Get currently-deployed etcd-events version + shell: "{{ docker_bin_dir }}/docker ps --filter='name={{ etcd_member_name }}-events' --format='{{ '{{ .Image }}' }}'" + register: etcd_events_current_docker_image + when: etcd_events_cluster_setup + +- name: Restart etcd if necessary + command: /bin/true + notify: Restart etcd + when: + - etcd_cluster_setup + - etcd_image_tag not in etcd_current_docker_image.stdout | default('') + +- name: Restart etcd-events if necessary + command: /bin/true + notify: Restart etcd-events + when: + - etcd_events_cluster_setup + - etcd_image_tag not in etcd_events_current_docker_image.stdout | default('') + +- name: Install etcd launch script + template: + src: etcd.j2 + dest: "{{ bin_dir }}/etcd" + owner: 'root' + mode: 0750 + backup: yes + when: etcd_cluster_setup + +- name: Install etcd-events launch script + template: + src: etcd-events.j2 + dest: "{{ bin_dir }}/etcd-events" + owner: 'root' + mode: 0750 + backup: yes + when: etcd_events_cluster_setup diff --git a/kubespray/roles/etcd/tasks/install_host.yml b/kubespray/roles/etcd/tasks/install_host.yml new file mode 100644 index 0000000..d4baa2a --- /dev/null +++ b/kubespray/roles/etcd/tasks/install_host.yml @@ -0,0 +1,31 @@ +--- +- name: Get currently-deployed etcd version + command: "{{ bin_dir }}/etcd --version" + register: etcd_current_host_version + # There's a chance this play could run before etcd is installed at all + ignore_errors: true + when: etcd_cluster_setup + +- name: Restart etcd if necessary + command: /bin/true + notify: Restart etcd + when: + - etcd_cluster_setup + - etcd_version.lstrip('v') not in etcd_current_host_version.stdout | default('') + +- name: Restart etcd-events if necessary + command: /bin/true + notify: Restart etcd-events + when: + - etcd_events_cluster_setup + - etcd_version.lstrip('v') not in etcd_current_host_version.stdout | default('') + +- name: Install | Copy etcd binary from download dir + copy: + src: "{{ local_release_dir }}/etcd-{{ etcd_version }}-linux-{{ host_architecture }}/{{ item }}" + dest: "{{ bin_dir }}/{{ item }}" + mode: 0755 + remote_src: yes + with_items: + - etcd + when: etcd_cluster_setup diff --git a/kubespray/roles/etcd/tasks/join_etcd-events_member.yml b/kubespray/roles/etcd/tasks/join_etcd-events_member.yml new file mode 100644 index 0000000..0fad331 --- /dev/null +++ b/kubespray/roles/etcd/tasks/join_etcd-events_member.yml @@ -0,0 +1,49 @@ +--- +- name: Join Member | Add member to etcd-events cluster + command: "{{ bin_dir }}/etcdctl member add {{ etcd_member_name }} --peer-urls={{ etcd_events_peer_url }}" + register: member_add_result + until: member_add_result.rc == 0 + retries: "{{ etcd_retries }}" + delay: "{{ retry_stagger | random + 3 }}" + environment: + ETCDCTL_API: "3" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + ETCDCTL_ENDPOINTS: "{{ etcd_events_access_addresses }}" + +- name: Join Member | Refresh etcd config + include_tasks: refresh_config.yml + vars: + # noqa: jinja[spacing] + etcd_events_peer_addresses: >- + {% for host in groups['etcd'] -%} + {%- if hostvars[host]['etcd_events_member_in_cluster'].rc == 0 -%} + {{ "etcd" + loop.index | string }}=https://{{ hostvars[host].etcd_events_access_address | default(hostvars[host].ip | default(fallback_ips[host])) }}:2382, + {%- endif -%} + {%- if loop.last -%} + {{ etcd_member_name }}={{ etcd_events_peer_url }} + {%- endif -%} + {%- endfor -%} + +- name: Join Member | Ensure member is in etcd-events cluster + shell: "set -o pipefail && {{ bin_dir }}/etcdctl member list | grep -w {{ etcd_events_access_address }} >/dev/null" + args: + executable: /bin/bash + register: etcd_events_member_in_cluster + changed_when: false + check_mode: no + tags: + - facts + environment: + ETCDCTL_API: "3" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + ETCDCTL_ENDPOINTS: "{{ etcd_events_access_addresses }}" + +- name: Configure | Ensure etcd-events is running + service: + name: etcd-events + state: started + enabled: yes diff --git a/kubespray/roles/etcd/tasks/join_etcd_member.yml b/kubespray/roles/etcd/tasks/join_etcd_member.yml new file mode 100644 index 0000000..ee77d4b --- /dev/null +++ b/kubespray/roles/etcd/tasks/join_etcd_member.yml @@ -0,0 +1,53 @@ +--- +- name: Join Member | Add member to etcd cluster + command: "{{ bin_dir }}/etcdctl member add {{ etcd_member_name }} --peer-urls={{ etcd_peer_url }}" + register: member_add_result + until: member_add_result.rc == 0 or 'Peer URLs already exists' in member_add_result.stderr + failed_when: member_add_result.rc != 0 and 'Peer URLs already exists' not in member_add_result.stderr + retries: "{{ etcd_retries }}" + delay: "{{ retry_stagger | random + 3 }}" + environment: + ETCDCTL_API: "3" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + ETCDCTL_ENDPOINTS: "{{ etcd_access_addresses }}" + +- name: Join Member | Refresh etcd config + include_tasks: refresh_config.yml + vars: + # noqa: jinja[spacing] + etcd_peer_addresses: >- + {% for host in groups['etcd'] -%} + {%- if hostvars[host]['etcd_member_in_cluster'].rc == 0 -%} + {{ "etcd" + loop.index | string }}=https://{{ hostvars[host].etcd_access_address | default(hostvars[host].ip | default(fallback_ips[host])) }}:2380, + {%- endif -%} + {%- if loop.last -%} + {{ etcd_member_name }}={{ etcd_peer_url }} + {%- endif -%} + {%- endfor -%} + +- name: Join Member | Ensure member is in etcd cluster + shell: "set -o pipefail && {{ bin_dir }}/etcdctl member list | grep -w {{ etcd_access_address }} >/dev/null" + args: + executable: /bin/bash + register: etcd_member_in_cluster + changed_when: false + check_mode: no + retries: "{{ etcd_retries }}" + delay: "{{ retry_stagger | random + 3 }}" + until: etcd_member_in_cluster.rc == 0 + tags: + - facts + environment: + ETCDCTL_API: "3" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + ETCDCTL_ENDPOINTS: "{{ etcd_access_addresses }}" + +- name: Configure | Ensure etcd is running + service: + name: etcd + state: started + enabled: yes diff --git a/kubespray/roles/etcd/tasks/main.yml b/kubespray/roles/etcd/tasks/main.yml new file mode 100644 index 0000000..40ca3de --- /dev/null +++ b/kubespray/roles/etcd/tasks/main.yml @@ -0,0 +1,96 @@ +--- +- name: Check etcd certs + include_tasks: check_certs.yml + when: cert_management == "script" + tags: + - etcd-secrets + - facts + +- name: Generate etcd certs + include_tasks: "gen_certs_script.yml" + when: + - cert_management | d('script') == "script" + tags: + - etcd-secrets + +- name: Trust etcd CA + include_tasks: upd_ca_trust.yml + when: + - inventory_hostname in groups['etcd'] | union(groups['kube_control_plane']) | unique | sort + tags: + - etcd-secrets + +- name: Trust etcd CA on nodes if needed + include_tasks: upd_ca_trust.yml + when: + - kube_network_plugin in ["calico", "flannel", "cilium"] or cilium_deploy_additionally | default(false) | bool + - kube_network_plugin != "calico" or calico_datastore == "etcd" + - inventory_hostname in groups['k8s_cluster'] + tags: + - etcd-secrets + +- name: "Gen_certs | Get etcd certificate serials" + command: "openssl x509 -in {{ etcd_cert_dir }}/node-{{ inventory_hostname }}.pem -noout -serial" + register: "etcd_client_cert_serial_result" + changed_when: false + check_mode: no + when: + - kube_network_plugin in ["calico", "flannel", "cilium"] or cilium_deploy_additionally | default(false) | bool + - kube_network_plugin != "calico" or calico_datastore == "etcd" + - inventory_hostname in groups['k8s_cluster'] + tags: + - master + - network + +- name: Set etcd_client_cert_serial + set_fact: + etcd_client_cert_serial: "{{ etcd_client_cert_serial_result.stdout.split('=')[1] }}" + when: + - kube_network_plugin in ["calico", "flannel", "cilium"] or cilium_deploy_additionally | default(false) | bool + - kube_network_plugin != "calico" or calico_datastore == "etcd" + - inventory_hostname in groups['k8s_cluster'] + tags: + - master + - network + +- name: Install etcdctl and etcdutl binary + import_role: + name: etcdctl_etcdutl + tags: + - etcdctl + - etcdutl + - upgrade + when: + - inventory_hostname in groups['etcd'] + - etcd_cluster_setup + +- name: Install etcd + include_tasks: "install_{{ etcd_deployment_type }}.yml" + when: is_etcd_master + tags: + - upgrade + +- name: Configure etcd + include_tasks: configure.yml + when: is_etcd_master + +- name: Refresh etcd config + include_tasks: refresh_config.yml + when: is_etcd_master + +- name: Restart etcd if certs changed + command: /bin/true + notify: Restart etcd + when: is_etcd_master and etcd_cluster_setup and etcd_secret_changed | default(false) + +- name: Restart etcd-events if certs changed + command: /bin/true + notify: Restart etcd + when: is_etcd_master and etcd_events_cluster_setup and etcd_secret_changed | default(false) + +# After etcd cluster is assembled, make sure that +# initial state of the cluster is in `existing` +# state instead of `new`. +- name: Refresh etcd config again for idempotency + include_tasks: refresh_config.yml + when: is_etcd_master diff --git a/kubespray/roles/etcd/tasks/refresh_config.yml b/kubespray/roles/etcd/tasks/refresh_config.yml new file mode 100644 index 0000000..d5e0045 --- /dev/null +++ b/kubespray/roles/etcd/tasks/refresh_config.yml @@ -0,0 +1,16 @@ +--- +- name: Refresh config | Create etcd config file + template: + src: etcd.env.j2 + dest: /etc/etcd.env + mode: 0640 + notify: Restart etcd + when: is_etcd_master and etcd_cluster_setup + +- name: Refresh config | Create etcd-events config file + template: + src: etcd-events.env.j2 + dest: /etc/etcd-events.env + mode: 0640 + notify: Restart etcd-events + when: is_etcd_master and etcd_events_cluster_setup diff --git a/kubespray/roles/etcd/tasks/upd_ca_trust.yml b/kubespray/roles/etcd/tasks/upd_ca_trust.yml new file mode 100644 index 0000000..22c5901 --- /dev/null +++ b/kubespray/roles/etcd/tasks/upd_ca_trust.yml @@ -0,0 +1,37 @@ +--- +- name: Gen_certs | target ca-certificate store file + set_fact: + ca_cert_path: |- + {% if ansible_os_family == "Debian" -%} + /usr/local/share/ca-certificates/etcd-ca.crt + {%- elif ansible_os_family == "RedHat" -%} + /etc/pki/ca-trust/source/anchors/etcd-ca.crt + {%- elif ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] -%} + /etc/ssl/certs/etcd-ca.pem + {%- elif ansible_os_family == "Suse" -%} + /etc/pki/trust/anchors/etcd-ca.pem + {%- elif ansible_os_family == "ClearLinux" -%} + /usr/share/ca-certs/etcd-ca.pem + {%- endif %} + tags: + - facts + +- name: Gen_certs | add CA to trusted CA dir + copy: + src: "{{ etcd_cert_dir }}/ca.pem" + dest: "{{ ca_cert_path }}" + remote_src: true + mode: 0640 + register: etcd_ca_cert + +- name: Gen_certs | update ca-certificates (Debian/Ubuntu/SUSE/Flatcar) # noqa no-handler + command: update-ca-certificates + when: etcd_ca_cert.changed and ansible_os_family in ["Debian", "Flatcar", "Flatcar Container Linux by Kinvolk", "Suse"] + +- name: Gen_certs | update ca-certificates (RedHat) # noqa no-handler + command: update-ca-trust extract + when: etcd_ca_cert.changed and ansible_os_family == "RedHat" + +- name: Gen_certs | update ca-certificates (ClearLinux) # noqa no-handler + command: clrtrust add "{{ ca_cert_path }}" + when: etcd_ca_cert.changed and ansible_os_family == "ClearLinux" diff --git a/kubespray/roles/etcd/templates/etcd-docker.service.j2 b/kubespray/roles/etcd/templates/etcd-docker.service.j2 new file mode 100644 index 0000000..4dfbd72 --- /dev/null +++ b/kubespray/roles/etcd/templates/etcd-docker.service.j2 @@ -0,0 +1,18 @@ +[Unit] +Description=etcd docker wrapper +Wants=docker.socket +After=docker.service + +[Service] +User=root +PermissionsStartOnly=true +EnvironmentFile=-/etc/etcd.env +ExecStart={{ bin_dir }}/etcd +ExecStartPre=-{{ docker_bin_dir }}/docker rm -f {{ etcd_member_name | default("etcd") }} +ExecStop={{ docker_bin_dir }}/docker stop {{ etcd_member_name | default("etcd") }} +Restart=always +RestartSec=15s +TimeoutStartSec=30s + +[Install] +WantedBy=multi-user.target diff --git a/kubespray/roles/etcd/templates/etcd-events-docker.service.j2 b/kubespray/roles/etcd/templates/etcd-events-docker.service.j2 new file mode 100644 index 0000000..271980a --- /dev/null +++ b/kubespray/roles/etcd/templates/etcd-events-docker.service.j2 @@ -0,0 +1,18 @@ +[Unit] +Description=etcd docker wrapper +Wants=docker.socket +After=docker.service + +[Service] +User=root +PermissionsStartOnly=true +EnvironmentFile=-/etc/etcd-events.env +ExecStart={{ bin_dir }}/etcd-events +ExecStartPre=-{{ docker_bin_dir }}/docker rm -f {{ etcd_member_name }}-events +ExecStop={{ docker_bin_dir }}/docker stop {{ etcd_member_name }}-events +Restart=always +RestartSec=15s +TimeoutStartSec=30s + +[Install] +WantedBy=multi-user.target diff --git a/kubespray/roles/etcd/templates/etcd-events-host.service.j2 b/kubespray/roles/etcd/templates/etcd-events-host.service.j2 new file mode 100644 index 0000000..6e0167a --- /dev/null +++ b/kubespray/roles/etcd/templates/etcd-events-host.service.j2 @@ -0,0 +1,16 @@ +[Unit] +Description=etcd +After=network.target + +[Service] +Type=notify +User=root +EnvironmentFile=/etc/etcd-events.env +ExecStart={{ bin_dir }}/etcd +NotifyAccess=all +Restart=always +RestartSec=10s +LimitNOFILE=40000 + +[Install] +WantedBy=multi-user.target diff --git a/kubespray/roles/etcd/templates/etcd-events.env.j2 b/kubespray/roles/etcd/templates/etcd-events.env.j2 new file mode 100644 index 0000000..3abefd6 --- /dev/null +++ b/kubespray/roles/etcd/templates/etcd-events.env.j2 @@ -0,0 +1,43 @@ +ETCD_DATA_DIR={{ etcd_events_data_dir }} +ETCD_ADVERTISE_CLIENT_URLS={{ etcd_events_client_url }} +ETCD_INITIAL_ADVERTISE_PEER_URLS={{ etcd_events_peer_url }} +ETCD_INITIAL_CLUSTER_STATE={% if etcd_events_cluster_is_healthy.rc == 0 | bool %}existing{% else %}new{% endif %} + +ETCD_METRICS={{ etcd_metrics }} +ETCD_LISTEN_CLIENT_URLS=https://{{ etcd_address }}:2383,https://127.0.0.1:2383 +ETCD_ELECTION_TIMEOUT={{ etcd_election_timeout }} +ETCD_HEARTBEAT_INTERVAL={{ etcd_heartbeat_interval }} +ETCD_INITIAL_CLUSTER_TOKEN=k8s_events_etcd +ETCD_LISTEN_PEER_URLS=https://{{ etcd_address }}:2382 +ETCD_NAME={{ etcd_member_name }}-events +ETCD_PROXY=off +ETCD_INITIAL_CLUSTER={{ etcd_events_peer_addresses }} +ETCD_AUTO_COMPACTION_RETENTION={{ etcd_compaction_retention }} +{% if etcd_snapshot_count is defined %} +ETCD_SNAPSHOT_COUNT={{ etcd_snapshot_count }} +{% endif %} +{% if etcd_quota_backend_bytes is defined %} +ETCD_QUOTA_BACKEND_BYTES={{ etcd_quota_backend_bytes }} +{% endif %} +{% if etcd_max_request_bytes is defined %} +ETCD_MAX_REQUEST_BYTES={{ etcd_max_request_bytes }} +{% endif %} + +# TLS settings +ETCD_TRUSTED_CA_FILE={{ etcd_cert_dir }}/ca.pem +ETCD_CERT_FILE={{ etcd_cert_dir }}/member-{{ inventory_hostname }}.pem +ETCD_KEY_FILE={{ etcd_cert_dir }}/member-{{ inventory_hostname }}-key.pem +ETCD_CLIENT_CERT_AUTH={{ etcd_secure_client | lower}} + +ETCD_PEER_TRUSTED_CA_FILE={{ etcd_cert_dir }}/ca.pem +ETCD_PEER_CERT_FILE={{ etcd_cert_dir }}/member-{{ inventory_hostname }}.pem +ETCD_PEER_KEY_FILE={{ etcd_cert_dir }}/member-{{ inventory_hostname }}-key.pem +ETCD_PEER_CLIENT_CERT_AUTH={{ etcd_peer_client_auth }} + +{% if etcd_tls_cipher_suites is defined %} +ETCD_CIPHER_SUITES={% for tls in etcd_tls_cipher_suites %}{{ tls }}{{ "," if not loop.last else "" }}{% endfor %} +{% endif %} + +{% for key, value in etcd_extra_vars.items() %} +{{ key }}={{ value }} +{% endfor %} diff --git a/kubespray/roles/etcd/templates/etcd-events.j2 b/kubespray/roles/etcd/templates/etcd-events.j2 new file mode 100644 index 0000000..b268479 --- /dev/null +++ b/kubespray/roles/etcd/templates/etcd-events.j2 @@ -0,0 +1,21 @@ +#!/bin/bash +{{ docker_bin_dir }}/docker run \ + --restart=on-failure:5 \ + --env-file=/etc/etcd-events.env \ + --net=host \ + -v /etc/ssl/certs:/etc/ssl/certs:ro \ + -v {{ etcd_cert_dir }}:{{ etcd_cert_dir }}:ro \ + -v {{ etcd_events_data_dir }}:{{ etcd_events_data_dir }}:rw \ + {% if etcd_memory_limit is defined %} + --memory={{ etcd_memory_limit|regex_replace('Mi', 'M') }} \ + {% endif %} + {% if etcd_cpu_limit is defined %} + --cpu-shares={{ etcd_cpu_limit|regex_replace('m', '') }} \ + {% endif %} + {% if etcd_blkio_weight is defined %} + --blkio-weight={{ etcd_blkio_weight }} \ + {% endif %} + --name={{ etcd_member_name }}-events \ + {{ etcd_image_repo }}:{{ etcd_image_tag }} \ + /usr/local/bin/etcd \ + "$@" diff --git a/kubespray/roles/etcd/templates/etcd-host.service.j2 b/kubespray/roles/etcd/templates/etcd-host.service.j2 new file mode 100644 index 0000000..6bba805 --- /dev/null +++ b/kubespray/roles/etcd/templates/etcd-host.service.j2 @@ -0,0 +1,16 @@ +[Unit] +Description=etcd +After=network.target + +[Service] +Type=notify +User=root +EnvironmentFile=/etc/etcd.env +ExecStart={{ bin_dir }}/etcd +NotifyAccess=all +Restart=always +RestartSec=10s +LimitNOFILE=40000 + +[Install] +WantedBy=multi-user.target diff --git a/kubespray/roles/etcd/templates/etcd.env.j2 b/kubespray/roles/etcd/templates/etcd.env.j2 new file mode 100644 index 0000000..2abd9cc --- /dev/null +++ b/kubespray/roles/etcd/templates/etcd.env.j2 @@ -0,0 +1,70 @@ +# Environment file for etcd {{ etcd_version }} +ETCD_DATA_DIR={{ etcd_data_dir }} +ETCD_ADVERTISE_CLIENT_URLS={{ etcd_client_url }} +ETCD_INITIAL_ADVERTISE_PEER_URLS={{ etcd_peer_url }} +ETCD_INITIAL_CLUSTER_STATE={% if etcd_cluster_is_healthy.rc == 0 | bool %}existing{% else %}new{% endif %} + +ETCD_METRICS={{ etcd_metrics }} +{% if etcd_listen_metrics_urls is defined %} +ETCD_LISTEN_METRICS_URLS={{ etcd_listen_metrics_urls }} +{% elif etcd_metrics_port is defined %} +ETCD_LISTEN_METRICS_URLS=http://{{ etcd_address }}:{{ etcd_metrics_port }},http://127.0.0.1:{{ etcd_metrics_port }} +{% endif %} +ETCD_LISTEN_CLIENT_URLS=https://{{ etcd_address }}:2379,https://127.0.0.1:2379 +ETCD_ELECTION_TIMEOUT={{ etcd_election_timeout }} +ETCD_HEARTBEAT_INTERVAL={{ etcd_heartbeat_interval }} +ETCD_INITIAL_CLUSTER_TOKEN=k8s_etcd +ETCD_LISTEN_PEER_URLS=https://{{ etcd_address }}:2380 +ETCD_NAME={{ etcd_member_name }} +ETCD_PROXY=off +ETCD_INITIAL_CLUSTER={{ etcd_peer_addresses }} +ETCD_AUTO_COMPACTION_RETENTION={{ etcd_compaction_retention }} +{% if etcd_snapshot_count is defined %} +ETCD_SNAPSHOT_COUNT={{ etcd_snapshot_count }} +{% endif %} +{% if etcd_quota_backend_bytes is defined %} +ETCD_QUOTA_BACKEND_BYTES={{ etcd_quota_backend_bytes }} +{% endif %} +{% if etcd_max_request_bytes is defined %} +ETCD_MAX_REQUEST_BYTES={{ etcd_max_request_bytes }} +{% endif %} +{% if etcd_log_level is defined %} +ETCD_LOG_LEVEL={{ etcd_log_level }} +{% endif %} +{% if etcd_max_snapshots is defined %} +ETCD_MAX_SNAPSHOTS={{ etcd_max_snapshots }} +{% endif %} +{% if etcd_max_wals is defined %} +ETCD_MAX_WALS={{ etcd_max_wals }} +{% endif %} +# Flannel need etcd v2 API +ETCD_ENABLE_V2=true + +# TLS settings +ETCD_TRUSTED_CA_FILE={{ etcd_cert_dir }}/ca.pem +ETCD_CERT_FILE={{ etcd_cert_dir }}/member-{{ inventory_hostname }}.pem +ETCD_KEY_FILE={{ etcd_cert_dir }}/member-{{ inventory_hostname }}-key.pem +ETCD_CLIENT_CERT_AUTH={{ etcd_secure_client | lower}} + +ETCD_PEER_TRUSTED_CA_FILE={{ etcd_cert_dir }}/ca.pem +ETCD_PEER_CERT_FILE={{ etcd_cert_dir }}/member-{{ inventory_hostname }}.pem +ETCD_PEER_KEY_FILE={{ etcd_cert_dir }}/member-{{ inventory_hostname }}-key.pem +ETCD_PEER_CLIENT_CERT_AUTH={{ etcd_peer_client_auth }} + +{% if etcd_tls_cipher_suites is defined %} +ETCD_CIPHER_SUITES={% for tls in etcd_tls_cipher_suites %}{{ tls }}{{ "," if not loop.last else "" }}{% endfor %} +{% endif %} + +{% for key, value in etcd_extra_vars.items() %} +{{ key }}={{ value }} +{% endfor %} + +# CLI settings +ETCDCTL_ENDPOINTS=https://127.0.0.1:2379 +ETCDCTL_CACERT={{ etcd_cert_dir }}/ca.pem +ETCDCTL_KEY={{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem +ETCDCTL_CERT={{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem + +# ETCD 3.5.x issue +# https://groups.google.com/a/kubernetes.io/g/dev/c/B7gJs88XtQc/m/rSgNOzV2BwAJ?utm_medium=email&utm_source=footer +ETCD_EXPERIMENTAL_INITIAL_CORRUPT_CHECK={{ etcd_experimental_initial_corrupt_check }} diff --git a/kubespray/roles/etcd/templates/etcd.j2 b/kubespray/roles/etcd/templates/etcd.j2 new file mode 100644 index 0000000..5374c70 --- /dev/null +++ b/kubespray/roles/etcd/templates/etcd.j2 @@ -0,0 +1,21 @@ +#!/bin/bash +{{ docker_bin_dir }}/docker run \ + --restart=on-failure:5 \ + --env-file=/etc/etcd.env \ + --net=host \ + -v /etc/ssl/certs:/etc/ssl/certs:ro \ + -v {{ etcd_cert_dir }}:{{ etcd_cert_dir }}:ro \ + -v {{ etcd_data_dir }}:{{ etcd_data_dir }}:rw \ +{% if etcd_memory_limit is defined %} + --memory={{ etcd_memory_limit|regex_replace('Mi', 'M') }} \ +{% endif %} +{% if etcd_cpu_limit is defined %} + --cpu-shares={{ etcd_cpu_limit|regex_replace('m', '') }} \ +{% endif %} +{% if etcd_blkio_weight is defined %} + --blkio-weight={{ etcd_blkio_weight }} \ +{% endif %} + --name={{ etcd_member_name | default("etcd") }} \ + {{ etcd_image_repo }}:{{ etcd_image_tag }} \ + /usr/local/bin/etcd \ + "$@" diff --git a/kubespray/roles/etcd/templates/make-ssl-etcd.sh.j2 b/kubespray/roles/etcd/templates/make-ssl-etcd.sh.j2 new file mode 100644 index 0000000..e0dde40 --- /dev/null +++ b/kubespray/roles/etcd/templates/make-ssl-etcd.sh.j2 @@ -0,0 +1,105 @@ +#!/bin/bash + +# Author: Smana smainklh@gmail.com +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +set -o errexit +set -o pipefail +usage() +{ + cat << EOF +Create self signed certificates + +Usage : $(basename $0) -f [-d ] + -h | --help : Show this message + -f | --config : Openssl configuration file + -d | --ssldir : Directory where the certificates will be installed + + ex : + $(basename $0) -f openssl.conf -d /srv/ssl +EOF +} + +# Options parsing +while (($#)); do + case "$1" in + -h | --help) usage; exit 0;; + -f | --config) CONFIG=${2}; shift 2;; + -d | --ssldir) SSLDIR="${2}"; shift 2;; + *) + usage + echo "ERROR : Unknown option" + exit 3 + ;; + esac +done + +if [ -z ${CONFIG} ]; then + echo "ERROR: the openssl configuration file is missing. option -f" + exit 1 +fi +if [ -z ${SSLDIR} ]; then + SSLDIR="/etc/ssl/etcd" +fi + +tmpdir=$(mktemp -d /tmp/etcd_cacert.XXXXXX) +trap 'rm -rf "${tmpdir}"' EXIT +cd "${tmpdir}" + +mkdir -p "${SSLDIR}" + +# Root CA +if [ -e "$SSLDIR/ca-key.pem" ]; then + # Reuse existing CA + cp $SSLDIR/{ca.pem,ca-key.pem} . +else + openssl genrsa -out ca-key.pem {{certificates_key_size}} > /dev/null 2>&1 + openssl req -x509 -new -nodes -key ca-key.pem -days {{certificates_duration}} -out ca.pem -subj "/CN=etcd-ca" > /dev/null 2>&1 +fi + +# ETCD member +if [ -n "$MASTERS" ]; then + for host in $MASTERS; do + cn="${host%%.*}" + # Member key + openssl genrsa -out member-${host}-key.pem {{certificates_key_size}} > /dev/null 2>&1 + openssl req -new -key member-${host}-key.pem -out member-${host}.csr -subj "/CN=etcd-member-${cn}" -config ${CONFIG} > /dev/null 2>&1 + openssl x509 -req -in member-${host}.csr -CA ca.pem -CAkey ca-key.pem -CAcreateserial -out member-${host}.pem -days {{certificates_duration}} -extensions ssl_client -extfile ${CONFIG} > /dev/null 2>&1 + + # Admin key + openssl genrsa -out admin-${host}-key.pem {{certificates_key_size}} > /dev/null 2>&1 + openssl req -new -key admin-${host}-key.pem -out admin-${host}.csr -subj "/CN=etcd-admin-${cn}" > /dev/null 2>&1 + openssl x509 -req -in admin-${host}.csr -CA ca.pem -CAkey ca-key.pem -CAcreateserial -out admin-${host}.pem -days {{certificates_duration}} -extensions ssl_client -extfile ${CONFIG} > /dev/null 2>&1 + done +fi + +# Node keys +if [ -n "$HOSTS" ]; then + for host in $HOSTS; do + cn="${host%%.*}" + openssl genrsa -out node-${host}-key.pem {{certificates_key_size}} > /dev/null 2>&1 + openssl req -new -key node-${host}-key.pem -out node-${host}.csr -subj "/CN=etcd-node-${cn}" > /dev/null 2>&1 + openssl x509 -req -in node-${host}.csr -CA ca.pem -CAkey ca-key.pem -CAcreateserial -out node-${host}.pem -days {{certificates_duration}} -extensions ssl_client -extfile ${CONFIG} > /dev/null 2>&1 + done +fi + +# Install certs +if [ -e "$SSLDIR/ca-key.pem" ]; then + # No pass existing CA + rm -f ca.pem ca-key.pem +fi + +if [ -n "$(ls -A *.pem)" ]; then + mv *.pem ${SSLDIR}/ +fi diff --git a/kubespray/roles/etcd/templates/openssl.conf.j2 b/kubespray/roles/etcd/templates/openssl.conf.j2 new file mode 100644 index 0000000..f6681a1 --- /dev/null +++ b/kubespray/roles/etcd/templates/openssl.conf.j2 @@ -0,0 +1,45 @@ +{% set counter = {'dns': 2,'ip': 1,} %}{% macro increment(dct, key, inc=1)%}{% if dct.update({key: dct[key] + inc}) %} {% endif %}{% endmacro %}[req] +req_extensions = v3_req +distinguished_name = req_distinguished_name + +[req_distinguished_name] + +[ v3_req ] +basicConstraints = CA:FALSE +keyUsage = nonRepudiation, digitalSignature, keyEncipherment +subjectAltName = @alt_names + +[ ssl_client ] +extendedKeyUsage = clientAuth, serverAuth +basicConstraints = CA:FALSE +subjectKeyIdentifier=hash +authorityKeyIdentifier=keyid,issuer +subjectAltName = @alt_names + +[ v3_ca ] +basicConstraints = CA:TRUE +keyUsage = nonRepudiation, digitalSignature, keyEncipherment +subjectAltName = @alt_names +authorityKeyIdentifier=keyid:always,issuer + +[alt_names] +DNS.1 = localhost +{% for host in groups['etcd'] %} +DNS.{{ counter["dns"] }} = {{ host }}{{ increment(counter, 'dns') }} +{% endfor %} +{% if apiserver_loadbalancer_domain_name is defined %} +DNS.{{ counter["dns"] }} = {{ apiserver_loadbalancer_domain_name }}{{ increment(counter, 'dns') }} +{% endif %} +{% for etcd_alt_name in etcd_cert_alt_names %} +DNS.{{ counter["dns"] }} = {{ etcd_alt_name }}{{ increment(counter, 'dns') }} +{% endfor %} +{% for host in groups['etcd'] %} +{% if hostvars[host]['access_ip'] is defined %} +IP.{{ counter["ip"] }} = {{ hostvars[host]['access_ip'] }}{{ increment(counter, 'ip') }} +{% endif %} +IP.{{ counter["ip"] }} = {{ hostvars[host]['ip'] | default(fallback_ips[host]) }}{{ increment(counter, 'ip') }} +{% endfor %} +{% for cert_alt_ip in etcd_cert_alt_ips %} +IP.{{ counter["ip"] }} = {{ cert_alt_ip }}{{ increment(counter, 'ip') }} +{% endfor %} +IP.{{ counter["ip"] }} = 127.0.0.1 diff --git a/kubespray/roles/etcdctl_etcdutl/tasks/main.yml b/kubespray/roles/etcdctl_etcdutl/tasks/main.yml new file mode 100644 index 0000000..39e87dc --- /dev/null +++ b/kubespray/roles/etcdctl_etcdutl/tasks/main.yml @@ -0,0 +1,45 @@ +--- +- name: Copy etcdctl and etcdutl binary from docker container + command: sh -c "{{ docker_bin_dir }}/docker rm -f etcdxtl-binarycopy; + {{ docker_bin_dir }}/docker create --name etcdxtl-binarycopy {{ etcd_image_repo }}:{{ etcd_image_tag }} && + {{ docker_bin_dir }}/docker cp etcdxtl-binarycopy:/usr/local/bin/{{ item }} {{ bin_dir }}/{{ item }} && + {{ docker_bin_dir }}/docker rm -f etcdxtl-binarycopy" + with_items: + - etcdctl + - etcdutl + register: etcdxtl_install_result + until: etcdxtl_install_result.rc == 0 + retries: "{{ etcd_retries }}" + delay: "{{ retry_stagger | random + 3 }}" + changed_when: false + when: container_manager == "docker" + +- name: Download etcd binary + include_tasks: "../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.etcd) }}" + when: container_manager in ['crio', 'containerd'] + +- name: Copy etcd binary + unarchive: + src: "{{ downloads.etcd.dest }}" + dest: "{{ local_release_dir }}/" + remote_src: yes + when: container_manager in ['crio', 'containerd'] + +- name: Copy etcdctl and etcdutl binary from download dir + copy: + src: "{{ local_release_dir }}/etcd-{{ etcd_version }}-linux-{{ host_architecture }}/{{ item }}" + dest: "{{ bin_dir }}/{{ item }}" + mode: 0755 + remote_src: yes + with_items: + - etcdctl + - etcdutl + when: container_manager in ['crio', 'containerd'] + +- name: Create etcdctl wrapper script + template: + src: etcdctl.sh.j2 + dest: "{{ bin_dir }}/etcdctl.sh" + mode: 0755 diff --git a/kubespray/roles/etcdctl_etcdutl/templates/etcdctl.sh.j2 b/kubespray/roles/etcdctl_etcdutl/templates/etcdctl.sh.j2 new file mode 100644 index 0000000..e4ddfec --- /dev/null +++ b/kubespray/roles/etcdctl_etcdutl/templates/etcdctl.sh.j2 @@ -0,0 +1,14 @@ +#!/bin/bash +# {{ ansible_managed }} +# example invocation: etcdctl.sh get --keys-only --from-key "" + +etcdctl \ +{% if etcd_deployment_type == "kubeadm" %} + --cacert {{ kube_cert_dir }}/etcd/ca.crt \ + --cert {{ kube_cert_dir }}/etcd/server.crt \ + --key {{ kube_cert_dir }}/etcd/server.key "$@" +{% else %} + --cacert {{ etcd_cert_dir }}/ca.pem \ + --cert {{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem \ + --key {{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem "$@" +{% endif %} diff --git a/kubespray/roles/helm-apps/README.md b/kubespray/roles/helm-apps/README.md new file mode 100644 index 0000000..8619688 --- /dev/null +++ b/kubespray/roles/helm-apps/README.md @@ -0,0 +1,39 @@ +Role Name +========= + +This role is intended to be used to fetch and deploy Helm Charts as part of +cluster installation or upgrading with kubespray. + +Requirements +------------ + +The role needs to be executed on a host with access to the Kubernetes API, and +with the helm binary in place. + +Role Variables +-------------- + +See meta/argument_specs.yml + +Playbook example: + +```yaml +--- +- hosts: kube_control_plane[0] + gather_facts: no + roles: + - name: helm-apps + releases: + - name: app + namespace: app + chart_ref: simple-app/simple-app + - name: app2 + namespace: app + chart_ref: simple-app/simple-app + wait_timeout: "10m" # override the same option in `release_common_opts` + repositories: "{{ repos }}" + - name: simple-app + url: "https://blog.leiwang.info/simple-app" + release_common_opts: "{{ helm_params }}" + wait_timeout: "5m" +``` diff --git a/kubespray/roles/helm-apps/meta/argument_specs.yml b/kubespray/roles/helm-apps/meta/argument_specs.yml new file mode 100644 index 0000000..d1be9a8 --- /dev/null +++ b/kubespray/roles/helm-apps/meta/argument_specs.yml @@ -0,0 +1,93 @@ +--- +argument_specs: + main: + short_description: Install a list of Helm charts. + options: + releases: + type: list + elements: dict + required: true + description: | + List of dictionaries passed as arguments to kubernetes.core.helm. + Arguments passed here will override those in `helm_settings`. For + structure of the dictionary, see the documentation for + kubernetes.core.helm ansible module. + options: + chart_ref: + type: path + required: true + chart_version: + type: str + name: + type: str + required: true + namespace: + type: str + required: true + values: + type: dict + # Possibly general options + create_namespace: + type: bool + chart_repo_url: + type: str + disable_hook: + type: bool + history_max: + type: int + purge: + type: bool + replace: + type: bool + skip_crds: + type: bool + wait: + type: bool + default: true + wait_timeout: + type: str + + repositories: + type: list + elements: dict + description: | + List of dictionaries passed as arguments to + kubernetes.core.helm_repository. + default: [] + options: + name: + type: str + required: true + password: + type: str + username: + type: str + url: + type: str + release_common_opts: + type: dict + description: | + Common arguments for every helm invocation. + default: {} + options: + create_namespace: + type: bool + default: true + chart_repo_url: + type: str + disable_hook: + type: bool + history_max: + type: int + purge: + type: bool + replace: + type: bool + skip_crds: + type: bool + wait: + type: bool + default: true + wait_timeout: + type: str + default: "5m" diff --git a/kubespray/roles/helm-apps/meta/main.yml b/kubespray/roles/helm-apps/meta/main.yml new file mode 100644 index 0000000..32ea6d4 --- /dev/null +++ b/kubespray/roles/helm-apps/meta/main.yml @@ -0,0 +1,3 @@ +--- +dependencies: + - role: kubernetes-apps/helm diff --git a/kubespray/roles/helm-apps/tasks/main.yml b/kubespray/roles/helm-apps/tasks/main.yml new file mode 100644 index 0000000..9515f16 --- /dev/null +++ b/kubespray/roles/helm-apps/tasks/main.yml @@ -0,0 +1,19 @@ +--- +- name: Add Helm repositories + kubernetes.core.helm_repository: "{{ helm_repository_defaults | combine(item) }}" # noqa args[module] + loop: "{{ repositories }}" + +- name: Update Helm repositories + kubernetes.core.helm: + state: absent + binary_path: "{{ bin_dir }}/helm" + release_name: dummy # trick needed to refresh in separate step + release_namespace: kube-system + update_repo_cache: true + when: + - repositories != [] + - helm_update + +- name: Install Helm Applications + kubernetes.core.helm: "{{ helm_defaults | combine(release_common_opts, item) }}" # noqa args[module] + loop: "{{ releases }}" diff --git a/kubespray/roles/helm-apps/vars/main.yml b/kubespray/roles/helm-apps/vars/main.yml new file mode 100644 index 0000000..bcce54a --- /dev/null +++ b/kubespray/roles/helm-apps/vars/main.yml @@ -0,0 +1,9 @@ +--- +helm_update: true + +helm_defaults: + atomic: true + binary_path: "{{ bin_dir }}/helm" + +helm_repository_defaults: + binary_path: "{{ bin_dir }}/helm" diff --git a/kubespray/roles/kubernetes-apps/ansible/defaults/main.yml b/kubespray/roles/kubernetes-apps/ansible/defaults/main.yml new file mode 100644 index 0000000..4de1fe9 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/defaults/main.yml @@ -0,0 +1,111 @@ +--- +# Limits for coredns + +# uncomment the line below to customize the DNS cpu limit value +# dns_cpu_limit: 300m +dns_memory_limit: 300Mi +dns_cpu_requests: 100m +dns_memory_requests: 70Mi +dns_min_replicas: "{{ [2, groups['k8s_cluster'] | length] | min }}" +dns_nodes_per_replica: 16 +dns_cores_per_replica: 256 +dns_prevent_single_point_failure: "{{ 'true' if dns_min_replicas | int > 1 else 'false' }}" +enable_coredns_reverse_dns_lookups: true +coredns_ordinal_suffix: "" +# dns_extra_tolerations: [{effect: NoSchedule, operator: "Exists"}] +coredns_deployment_nodeselector: "kubernetes.io/os: linux" +coredns_default_zone_cache_block: | + cache 30 +# coredns_additional_configs adds any extra configuration to coredns +# coredns_additional_configs: | +# whoami +# local + +# coredns_rewrite_block: | +# rewrite stop { +# name regex (.*)\.my\.domain {1}.svc.cluster.local +# answer name (.*)\.svc\.cluster\.local {1}.my.domain +# } + + +# dns_upstream_forward_extra_opts apply to coredns forward section as well as nodelocaldns upstream target forward section +# dns_upstream_forward_extra_opts: +# policy: sequential + +# Apply extra options to coredns kubernetes plugin +# coredns_kubernetes_extra_opts: +# - 'fallthrough example.local' + +# nodelocaldns +nodelocaldns_cpu_requests: 100m +nodelocaldns_memory_limit: 200Mi +nodelocaldns_memory_requests: 70Mi +nodelocaldns_ds_nodeselector: "kubernetes.io/os: linux" +nodelocaldns_prometheus_port: 9253 +nodelocaldns_secondary_prometheus_port: 9255 + +# Limits for dns-autoscaler +dns_autoscaler_cpu_requests: 20m +dns_autoscaler_memory_requests: 10Mi +dns_autoscaler_deployment_nodeselector: "kubernetes.io/os: linux" +# dns_autoscaler_extra_tolerations: [{effect: NoSchedule, operator: "Exists"}] + +# etcd metrics +# etcd_metrics_service_labels: +# k8s-app: etcd +# app.kubernetes.io/managed-by: Kubespray +# app: kube-prometheus-stack-kube-etcd +# release: prometheus-stack + +# Netchecker +deploy_netchecker: false +netchecker_port: 31081 +agent_report_interval: 15 +netcheck_namespace: default + +# Limits for netchecker apps +netchecker_agent_cpu_limit: 30m +netchecker_agent_memory_limit: 100M +netchecker_agent_cpu_requests: 15m +netchecker_agent_memory_requests: 64M +netchecker_server_cpu_limit: 100m +netchecker_server_memory_limit: 256M +netchecker_server_cpu_requests: 50m +netchecker_server_memory_requests: 64M +netchecker_etcd_cpu_limit: 200m +netchecker_etcd_memory_limit: 256M +netchecker_etcd_cpu_requests: 100m +netchecker_etcd_memory_requests: 128M + +# SecurityContext when PodSecurityPolicy is enabled +netchecker_agent_user: 1000 +netchecker_server_user: 1000 +netchecker_agent_group: 1000 +netchecker_server_group: 1000 + +# Dashboard +dashboard_replicas: 1 + +# Namespace for dashboard +dashboard_namespace: kube-system + +# Limits for dashboard +dashboard_cpu_limit: 100m +dashboard_memory_limit: 256M +dashboard_cpu_requests: 50m +dashboard_memory_requests: 64M + +# Set dashboard_use_custom_certs to true if overriding dashboard_certs_secret_name with a secret that +# contains dashboard_tls_key_file and dashboard_tls_cert_file instead of using the initContainer provisioned certs +dashboard_use_custom_certs: false +dashboard_certs_secret_name: kubernetes-dashboard-certs +dashboard_tls_key_file: dashboard.key +dashboard_tls_cert_file: dashboard.crt +dashboard_master_toleration: true + +# Override dashboard default settings +dashboard_token_ttl: 900 +dashboard_skip_login: false + +# Policy Controllers +# policy_controller_extra_tolerations: [{effect: NoSchedule, operator: "Exists"}] diff --git a/kubespray/roles/kubernetes-apps/ansible/tasks/cleanup_dns.yml b/kubespray/roles/kubernetes-apps/ansible/tasks/cleanup_dns.yml new file mode 100644 index 0000000..fef5246 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/tasks/cleanup_dns.yml @@ -0,0 +1,44 @@ +--- +- name: Kubernetes Apps | Register coredns deployment annotation `createdby` + command: "{{ kubectl }} get deploy -n kube-system coredns -o jsonpath='{ .spec.template.metadata.annotations.createdby }'" + register: createdby_annotation_deploy + changed_when: false + check_mode: false + ignore_errors: true # noqa ignore-errors + when: + - dns_mode in ['coredns', 'coredns_dual'] + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Apps | Register coredns service annotation `createdby` + command: "{{ kubectl }} get svc -n kube-system coredns -o jsonpath='{ .metadata.annotations.createdby }'" + register: createdby_annotation_svc + changed_when: false + check_mode: false + ignore_errors: true # noqa ignore-errors + when: + - dns_mode in ['coredns', 'coredns_dual'] + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Apps | Delete kubeadm CoreDNS + kube: + name: "coredns" + namespace: "kube-system" + kubectl: "{{ bin_dir }}/kubectl" + resource: "deploy" + state: absent + when: + - dns_mode in ['coredns', 'coredns_dual'] + - inventory_hostname == groups['kube_control_plane'][0] + - createdby_annotation_deploy.stdout != 'kubespray' + +- name: Kubernetes Apps | Delete kubeadm Kube-DNS service + kube: + name: "kube-dns" + namespace: "kube-system" + kubectl: "{{ bin_dir }}/kubectl" + resource: "svc" + state: absent + when: + - dns_mode in ['coredns', 'coredns_dual'] + - inventory_hostname == groups['kube_control_plane'][0] + - createdby_annotation_svc.stdout != 'kubespray' diff --git a/kubespray/roles/kubernetes-apps/ansible/tasks/coredns.yml b/kubespray/roles/kubernetes-apps/ansible/tasks/coredns.yml new file mode 100644 index 0000000..d8f8547 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/tasks/coredns.yml @@ -0,0 +1,44 @@ +--- +- name: Kubernetes Apps | Lay Down CoreDNS templates + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + loop: + - { name: coredns, file: coredns-clusterrole.yml, type: clusterrole } + - { name: coredns, file: coredns-clusterrolebinding.yml, type: clusterrolebinding } + - { name: coredns, file: coredns-config.yml, type: configmap } + - { name: coredns, file: coredns-deployment.yml, type: deployment } + - { name: coredns, file: coredns-sa.yml, type: sa } + - { name: coredns, file: coredns-svc.yml, type: svc } + - { name: dns-autoscaler, file: dns-autoscaler.yml, type: deployment } + - { name: dns-autoscaler, file: dns-autoscaler-clusterrole.yml, type: clusterrole } + - { name: dns-autoscaler, file: dns-autoscaler-clusterrolebinding.yml, type: clusterrolebinding } + - { name: dns-autoscaler, file: dns-autoscaler-sa.yml, type: sa } + register: coredns_manifests + vars: + clusterIP: "{{ skydns_server }}" + when: + - dns_mode in ['coredns', 'coredns_dual'] + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - coredns + +- name: Kubernetes Apps | Lay Down Secondary CoreDNS Template + template: + src: "{{ item.src }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - { name: coredns, src: coredns-deployment.yml, file: coredns-deployment-secondary.yml, type: deployment } + - { name: coredns, src: coredns-svc.yml, file: coredns-svc-secondary.yml, type: svc } + - { name: dns-autoscaler, src: dns-autoscaler.yml, file: coredns-autoscaler-secondary.yml, type: deployment } + register: coredns_secondary_manifests + vars: + clusterIP: "{{ skydns_server_secondary }}" + coredns_ordinal_suffix: "-secondary" + when: + - dns_mode == 'coredns_dual' + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - coredns diff --git a/kubespray/roles/kubernetes-apps/ansible/tasks/dashboard.yml b/kubespray/roles/kubernetes-apps/ansible/tasks/dashboard.yml new file mode 100644 index 0000000..480b3db --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/tasks/dashboard.yml @@ -0,0 +1,21 @@ +--- +- name: Kubernetes Apps | Lay down dashboard template + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - { file: dashboard.yml, type: deploy, name: kubernetes-dashboard } + register: manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Apps | Start dashboard + kube: + name: "{{ item.item.name }}" + namespace: "{{ dashboard_namespace }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: "{{ manifests.results }}" + when: inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/ansible/tasks/etcd_metrics.yml b/kubespray/roles/kubernetes-apps/ansible/tasks/etcd_metrics.yml new file mode 100644 index 0000000..548de89 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/tasks/etcd_metrics.yml @@ -0,0 +1,22 @@ +--- +- name: Kubernetes Apps | Lay down etcd_metrics templates + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - { file: etcd_metrics-endpoints.yml, type: endpoints, name: etcd-metrics } + - { file: etcd_metrics-service.yml, type: service, name: etcd-metrics } + register: manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Apps | Start etcd_metrics + kube: + name: "{{ item.item.name }}" + namespace: kube-system + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: "{{ manifests.results }}" + when: inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/ansible/tasks/main.yml b/kubespray/roles/kubernetes-apps/ansible/tasks/main.yml new file mode 100644 index 0000000..4a0180e --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/tasks/main.yml @@ -0,0 +1,82 @@ +--- +- name: Kubernetes Apps | Wait for kube-apiserver + uri: + url: "{{ kube_apiserver_endpoint }}/healthz" + validate_certs: no + client_cert: "{{ kube_apiserver_client_cert }}" + client_key: "{{ kube_apiserver_client_key }}" + register: result + until: result.status == 200 + retries: 20 + delay: 1 + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Apps | Cleanup DNS + import_tasks: cleanup_dns.yml + when: + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - upgrade + - coredns + - nodelocaldns + +- name: Kubernetes Apps | CoreDNS + import_tasks: "coredns.yml" + when: + - dns_mode in ['coredns', 'coredns_dual'] + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - coredns + +- name: Kubernetes Apps | nodelocalDNS + import_tasks: "nodelocaldns.yml" + when: + - enable_nodelocaldns + - inventory_hostname == groups['kube_control_plane'] | first + tags: + - nodelocaldns + +- name: Kubernetes Apps | Start Resources + kube: + name: "{{ item.item.name }}" + namespace: "kube-system" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ coredns_manifests.results | default({}) }}" + - "{{ coredns_secondary_manifests.results | default({}) }}" + - "{{ nodelocaldns_manifests.results | default({}) }}" + - "{{ nodelocaldns_second_manifests.results | default({}) }}" + when: + - dns_mode != 'none' + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + register: resource_result + until: resource_result is succeeded + retries: 4 + delay: 5 + tags: + - coredns + - nodelocaldns + loop_control: + label: "{{ item.item.file }}" + +- name: Kubernetes Apps | Etcd metrics endpoints + import_tasks: etcd_metrics.yml + when: etcd_metrics_port is defined and etcd_metrics_service_labels is defined + tags: + - etcd_metrics + +- name: Kubernetes Apps | Netchecker + import_tasks: netchecker.yml + when: deploy_netchecker + tags: + - netchecker + +- name: Kubernetes Apps | Dashboard + import_tasks: dashboard.yml + when: dashboard_enabled + tags: + - dashboard diff --git a/kubespray/roles/kubernetes-apps/ansible/tasks/netchecker.yml b/kubespray/roles/kubernetes-apps/ansible/tasks/netchecker.yml new file mode 100644 index 0000000..b83fd33 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/tasks/netchecker.yml @@ -0,0 +1,56 @@ +--- +- name: Kubernetes Apps | Check AppArmor status + command: which apparmor_parser + register: apparmor_status + when: + - inventory_hostname == groups['kube_control_plane'][0] + failed_when: false + +- name: Kubernetes Apps | Set apparmor_enabled + set_fact: + apparmor_enabled: "{{ apparmor_status.rc == 0 }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Apps | Netchecker Templates list + set_fact: + netchecker_templates: + - {file: netchecker-ns.yml, type: ns, name: netchecker-namespace} + - {file: netchecker-agent-sa.yml, type: sa, name: netchecker-agent} + - {file: netchecker-agent-ds.yml, type: ds, name: netchecker-agent} + - {file: netchecker-agent-hostnet-ds.yml, type: ds, name: netchecker-agent-hostnet} + - {file: netchecker-server-sa.yml, type: sa, name: netchecker-server} + - {file: netchecker-server-clusterrole.yml, type: clusterrole, name: netchecker-server} + - {file: netchecker-server-clusterrolebinding.yml, type: clusterrolebinding, name: netchecker-server} + - {file: netchecker-server-deployment.yml, type: deployment, name: netchecker-server} + - {file: netchecker-server-svc.yml, type: svc, name: netchecker-service} + netchecker_templates_for_psp: + - {file: netchecker-agent-hostnet-psp.yml, type: podsecuritypolicy, name: netchecker-agent-hostnet-policy} + - {file: netchecker-agent-hostnet-clusterrole.yml, type: clusterrole, name: netchecker-agent} + - {file: netchecker-agent-hostnet-clusterrolebinding.yml, type: clusterrolebinding, name: netchecker-agent} + +- name: Kubernetes Apps | Append extra templates to Netchecker Templates list for PodSecurityPolicy + set_fact: + netchecker_templates: "{{ netchecker_templates_for_psp + netchecker_templates }}" + when: podsecuritypolicy_enabled + +- name: Kubernetes Apps | Lay Down Netchecker Template + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: "{{ netchecker_templates }}" + register: manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Apps | Start Netchecker Resources + kube: + name: "{{ item.item.name }}" + namespace: "{{ netcheck_namespace }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: "{{ manifests.results }}" + when: inventory_hostname == groups['kube_control_plane'][0] and not item is skipped diff --git a/kubespray/roles/kubernetes-apps/ansible/tasks/nodelocaldns.yml b/kubespray/roles/kubernetes-apps/ansible/tasks/nodelocaldns.yml new file mode 100644 index 0000000..b438afb --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/tasks/nodelocaldns.yml @@ -0,0 +1,79 @@ +--- +- name: Kubernetes Apps | set up necessary nodelocaldns parameters + set_fact: + # noqa: jinja[spacing] + primaryClusterIP: >- + {%- if dns_mode in ['coredns', 'coredns_dual'] -%} + {{ skydns_server }} + {%- elif dns_mode == 'manual' -%} + {{ manual_dns_server }} + {%- endif -%} + secondaryclusterIP: "{{ skydns_server_secondary }}" + when: + - enable_nodelocaldns + - inventory_hostname == groups['kube_control_plane'] | first + tags: + - nodelocaldns + - coredns + +- name: Kubernetes Apps | Lay Down nodelocaldns Template + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - { name: nodelocaldns, file: nodelocaldns-config.yml, type: configmap } + - { name: nodelocaldns, file: nodelocaldns-sa.yml, type: sa } + - { name: nodelocaldns, file: nodelocaldns-daemonset.yml, type: daemonset } + register: nodelocaldns_manifests + vars: + # noqa: jinja[spacing] + forwardTarget: >- + {%- if secondaryclusterIP is defined and dns_mode == 'coredns_dual' -%} + {{ primaryClusterIP }} {{ secondaryclusterIP }} + {%- else -%} + {{ primaryClusterIP }} + {%- endif -%} + upstreamForwardTarget: >- + {%- if upstream_dns_servers is defined and upstream_dns_servers | length > 0 -%} + {{ upstream_dns_servers | join(' ') }} + {%- else -%} + /etc/resolv.conf + {%- endif -%} + when: + - enable_nodelocaldns + - inventory_hostname == groups['kube_control_plane'] | first + tags: + - nodelocaldns + - coredns + +- name: Kubernetes Apps | Lay Down nodelocaldns-secondary Template + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - { name: nodelocaldns, file: nodelocaldns-second-daemonset.yml, type: daemonset } + register: nodelocaldns_second_manifests + vars: + # noqa: jinja[spacing] + forwardTarget: >- + {%- if secondaryclusterIP is defined and dns_mode == 'coredns_dual' -%} + {{ primaryClusterIP }} {{ secondaryclusterIP }} + {%- else -%} + {{ primaryClusterIP }} + {%- endif -%} + # noqa: jinja[spacing] + upstreamForwardTarget: >- + {%- if upstream_dns_servers is defined and upstream_dns_servers | length > 0 -%} + {{ upstream_dns_servers | join(' ') }} + {%- else -%} + /etc/resolv.conf + {%- endif -%} + when: + - enable_nodelocaldns + - enable_nodelocaldns_secondary + - inventory_hostname == groups['kube_control_plane'] | first + tags: + - nodelocaldns + - coredns diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/coredns-clusterrole.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/coredns-clusterrole.yml.j2 new file mode 100644 index 0000000..d5f91ed --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/coredns-clusterrole.yml.j2 @@ -0,0 +1,32 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + kubernetes.io/bootstrapping: rbac-defaults + addonmanager.kubernetes.io/mode: Reconcile + name: system:coredns +rules: +- apiGroups: + - "" + resources: + - endpoints + - services + - pods + - namespaces + verbs: + - list + - watch +- apiGroups: + - "" + resources: + - nodes + verbs: + - get +- apiGroups: + - discovery.k8s.io + resources: + - endpointslices + verbs: + - list + - watch diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/coredns-clusterrolebinding.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/coredns-clusterrolebinding.yml.j2 new file mode 100644 index 0000000..af7f684 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/coredns-clusterrolebinding.yml.j2 @@ -0,0 +1,18 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + annotations: + rbac.authorization.kubernetes.io/autoupdate: "true" + labels: + kubernetes.io/bootstrapping: rbac-defaults + addonmanager.kubernetes.io/mode: EnsureExists + name: system:coredns +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: system:coredns +subjects: + - kind: ServiceAccount + name: coredns + namespace: kube-system diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/coredns-config.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/coredns-config.yml.j2 new file mode 100644 index 0000000..7a06023 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/coredns-config.yml.j2 @@ -0,0 +1,85 @@ +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: coredns + namespace: kube-system + labels: + addonmanager.kubernetes.io/mode: EnsureExists +data: + Corefile: | +{% if coredns_external_zones is defined and coredns_external_zones | length > 0 %} +{% for block in coredns_external_zones %} + {{ block['zones'] | join(' ') }} { + log + errors +{% if block['rewrite'] is defined and block['rewrite'] | length > 0 %} +{% for rewrite_match in block['rewrite'] %} + rewrite {{ rewrite_match }} +{% endfor %} +{% endif %} + forward . {{ block['nameservers'] | join(' ') }} + loadbalance + cache {{ block['cache'] | default(5) }} + reload +{% if dns_etchosts | default(None) %} + hosts /etc/coredns/hosts { + fallthrough + } +{% endif %} + } +{% endfor %} +{% endif %} + .:53 { + {% if coredns_additional_configs is defined %} + {{ coredns_additional_configs | indent(width=8, first=False) }} + {% endif %} + errors + health { + lameduck 5s + } +{% if coredns_rewrite_block is defined %} + {{ coredns_rewrite_block | indent(width=8, first=False) }} +{% endif %} + ready + kubernetes {{ dns_domain }} {% if coredns_kubernetes_extra_domains is defined %}{{ coredns_kubernetes_extra_domains }} {% endif %}{% if enable_coredns_reverse_dns_lookups %}in-addr.arpa ip6.arpa {% endif %}{ + pods insecure +{% if enable_coredns_k8s_endpoint_pod_names %} + endpoint_pod_names +{% endif %} +{% if enable_coredns_reverse_dns_lookups %} + fallthrough in-addr.arpa ip6.arpa +{% endif %} +{% if coredns_kubernetes_extra_opts is defined %} +{% for opt in coredns_kubernetes_extra_opts %} + {{ opt }} +{% endfor %} +{% endif %} + } + prometheus :9153 + forward . {{ upstream_dns_servers | join(' ') if upstream_dns_servers is defined and upstream_dns_servers | length > 0 else '/etc/resolv.conf' }} { + prefer_udp + max_concurrent 1000 +{% if dns_upstream_forward_extra_opts is defined %} +{% for optname, optvalue in dns_upstream_forward_extra_opts.items() %} + {{ optname }} {{ optvalue }} +{% endfor %} +{% endif %} + } +{% if enable_coredns_k8s_external %} + k8s_external {{ coredns_k8s_external_zone }} +{% endif %} + {{ coredns_default_zone_cache_block | indent(width=8, first=False) }} + loop + reload + loadbalance +{% if dns_etchosts | default(None) %} + hosts /etc/coredns/hosts { + fallthrough + } +{% endif %} + } +{% if dns_etchosts | default(None) %} + hosts: | + {{ dns_etchosts | indent(width=4, first=False) }} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/coredns-deployment.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/coredns-deployment.yml.j2 new file mode 100644 index 0000000..6cb7604 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/coredns-deployment.yml.j2 @@ -0,0 +1,124 @@ +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + name: "coredns{{ coredns_ordinal_suffix }}" + namespace: kube-system + labels: + k8s-app: "kube-dns{{ coredns_ordinal_suffix }}" + addonmanager.kubernetes.io/mode: Reconcile + kubernetes.io/name: "coredns{{ coredns_ordinal_suffix }}" +spec: + strategy: + type: RollingUpdate + rollingUpdate: + maxUnavailable: 0 + maxSurge: 10% + selector: + matchLabels: + k8s-app: kube-dns{{ coredns_ordinal_suffix }} + template: + metadata: + labels: + k8s-app: kube-dns{{ coredns_ordinal_suffix }} + annotations: + createdby: 'kubespray' + spec: + securityContext: + seccompProfile: + type: RuntimeDefault + nodeSelector: + {{ coredns_deployment_nodeselector }} + priorityClassName: system-cluster-critical + serviceAccountName: coredns + tolerations: + - key: node-role.kubernetes.io/master + effect: NoSchedule + - key: node-role.kubernetes.io/control-plane + effect: NoSchedule +{% if dns_extra_tolerations is defined %} + {{ dns_extra_tolerations | list | to_nice_yaml(indent=2) | indent(8) }} +{% endif %} + affinity: + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - topologyKey: "kubernetes.io/hostname" + labelSelector: + matchLabels: + k8s-app: kube-dns{{ coredns_ordinal_suffix }} + nodeAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - weight: 100 + preference: + matchExpressions: + - key: node-role.kubernetes.io/control-plane + operator: In + values: + - "" + containers: + - name: coredns + image: "{{ coredns_image_repo }}:{{ coredns_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + resources: + # TODO: Set memory limits when we've profiled the container for large + # clusters, then set request = limit to keep this container in + # guaranteed class. Currently, this container falls into the + # "burstable" category so the kubelet doesn't backoff from restarting it. + limits: +{% if dns_cpu_limit is defined %} + cpu: {{ dns_cpu_limit }} +{% endif %} + memory: {{ dns_memory_limit }} + requests: + cpu: {{ dns_cpu_requests }} + memory: {{ dns_memory_requests }} + args: [ "-conf", "/etc/coredns/Corefile" ] + volumeMounts: + - name: config-volume + mountPath: /etc/coredns + ports: + - containerPort: 53 + name: dns + protocol: UDP + - containerPort: 53 + name: dns-tcp + protocol: TCP + - containerPort: 9153 + name: metrics + protocol: TCP + securityContext: + allowPrivilegeEscalation: false + capabilities: + add: + - NET_BIND_SERVICE + drop: + - all + readOnlyRootFilesystem: true + livenessProbe: + httpGet: + path: /health + port: 8080 + scheme: HTTP + timeoutSeconds: 5 + successThreshold: 1 + failureThreshold: 10 + readinessProbe: + httpGet: + path: /ready + port: 8181 + scheme: HTTP + timeoutSeconds: 5 + successThreshold: 1 + failureThreshold: 10 + dnsPolicy: Default + volumes: + - name: config-volume + configMap: + name: coredns + items: + - key: Corefile + path: Corefile +{% if dns_etchosts | default(None) %} + - key: hosts + path: hosts +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/coredns-sa.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/coredns-sa.yml.j2 new file mode 100644 index 0000000..64d9c4d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/coredns-sa.yml.j2 @@ -0,0 +1,9 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: coredns + namespace: kube-system + labels: + kubernetes.io/cluster-service: "true" + addonmanager.kubernetes.io/mode: Reconcile diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/coredns-svc.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/coredns-svc.yml.j2 new file mode 100644 index 0000000..0e051c3 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/coredns-svc.yml.j2 @@ -0,0 +1,28 @@ +--- +apiVersion: v1 +kind: Service +metadata: + name: coredns{{ coredns_ordinal_suffix }} + namespace: kube-system + labels: + k8s-app: kube-dns{{ coredns_ordinal_suffix }} + kubernetes.io/name: "coredns{{ coredns_ordinal_suffix }}" + addonmanager.kubernetes.io/mode: Reconcile + annotations: + prometheus.io/port: "9153" + prometheus.io/scrape: "true" + createdby: 'kubespray' +spec: + selector: + k8s-app: kube-dns{{ coredns_ordinal_suffix }} + clusterIP: {{ clusterIP }} + ports: + - name: dns + port: 53 + protocol: UDP + - name: dns-tcp + port: 53 + protocol: TCP + - name: metrics + port: 9153 + protocol: TCP diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/dashboard.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/dashboard.yml.j2 new file mode 100644 index 0000000..b0c3419 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/dashboard.yml.j2 @@ -0,0 +1,339 @@ +# Copyright 2017 The Kubernetes Authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +# Configuration to deploy release version of the Dashboard UI compatible with +# Kubernetes 1.8. +# +# Example usage: kubectl create -f + +{% if dashboard_namespace != "kube-system" %} +--- +apiVersion: v1 +kind: Namespace +metadata: + name: {{ dashboard_namespace }} + labels: + name: {{ dashboard_namespace }} +{% endif %} + +--- +# ------------------- Dashboard Secrets ------------------- # +apiVersion: v1 +kind: Secret +metadata: + labels: + k8s-app: kubernetes-dashboard + name: kubernetes-dashboard-certs + namespace: {{ dashboard_namespace }} +type: Opaque + +--- +apiVersion: v1 +kind: Secret +metadata: + labels: + k8s-app: kubernetes-dashboard + name: kubernetes-dashboard-csrf + namespace: {{ dashboard_namespace }} +type: Opaque +data: + csrf: "" + +--- +apiVersion: v1 +kind: Secret +metadata: + labels: + k8s-app: kubernetes-dashboard + name: kubernetes-dashboard-key-holder + namespace: {{ dashboard_namespace }} +type: Opaque + +--- +# ------------------- Dashboard ConfigMap ------------------- # +kind: ConfigMap +apiVersion: v1 +metadata: + labels: + k8s-app: kubernetes-dashboard + name: kubernetes-dashboard-settings + namespace: {{ dashboard_namespace }} + +--- +# ------------------- Dashboard Service Account ------------------- # + +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + k8s-app: kubernetes-dashboard + name: kubernetes-dashboard + namespace: {{ dashboard_namespace }} + +--- +# ------------------- Dashboard Role & Role Binding ------------------- # +kind: Role +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: + k8s-app: kubernetes-dashboard + name: kubernetes-dashboard + namespace: {{ dashboard_namespace }} +rules: + # Allow Dashboard to get, update and delete Dashboard exclusive secrets. + - apiGroups: [""] + resources: ["secrets"] + resourceNames: ["kubernetes-dashboard-key-holder", "kubernetes-dashboard-certs", "kubernetes-dashboard-csrf"] + verbs: ["get", "update", "delete"] + # Allow Dashboard to get and update 'kubernetes-dashboard-settings' config map. + - apiGroups: [""] + resources: ["configmaps"] + resourceNames: ["kubernetes-dashboard-settings"] + verbs: ["get", "update"] + # Allow Dashboard to get metrics. + - apiGroups: [""] + resources: ["services"] + resourceNames: ["heapster", "dashboard-metrics-scraper"] + verbs: ["proxy"] + - apiGroups: [""] + resources: ["services/proxy"] + resourceNames: ["heapster", "http:heapster:", "https:heapster:", "dashboard-metrics-scraper", "http:dashboard-metrics-scraper"] + verbs: ["get"] + +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + labels: + k8s-app: kubernetes-dashboard + name: kubernetes-dashboard + namespace: {{ dashboard_namespace }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: kubernetes-dashboard +subjects: + - kind: ServiceAccount + name: kubernetes-dashboard + namespace: {{ dashboard_namespace }} + +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: kubernetes-dashboard +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: kubernetes-dashboard +subjects: + - kind: ServiceAccount + name: kubernetes-dashboard + namespace: {{ dashboard_namespace }} + +--- +# ------------------- Dashboard Deployment ------------------- # + +kind: Deployment +apiVersion: apps/v1 +metadata: + labels: + k8s-app: kubernetes-dashboard + name: kubernetes-dashboard + namespace: {{ dashboard_namespace }} +spec: + replicas: {{ dashboard_replicas }} + revisionHistoryLimit: 10 + selector: + matchLabels: + k8s-app: kubernetes-dashboard + template: + metadata: + labels: + k8s-app: kubernetes-dashboard + spec: + securityContext: + seccompProfile: + type: RuntimeDefault + priorityClassName: system-cluster-critical + containers: + - name: kubernetes-dashboard + image: {{ dashboard_image_repo }}:{{ dashboard_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + resources: + limits: + cpu: {{ dashboard_cpu_limit }} + memory: {{ dashboard_memory_limit }} + requests: + cpu: {{ dashboard_cpu_requests }} + memory: {{ dashboard_memory_requests }} + ports: + - containerPort: 8443 + protocol: TCP + args: + - --namespace={{ dashboard_namespace }} +{% if dashboard_use_custom_certs %} + - --tls-key-file={{ dashboard_tls_key_file }} + - --tls-cert-file={{ dashboard_tls_cert_file }} +{% else %} + - --auto-generate-certificates +{% endif %} +{% if dashboard_skip_login %} + - --enable-skip-login +{% endif %} + - --authentication-mode=token + # Uncomment the following line to manually specify Kubernetes API server Host + # If not specified, Dashboard will attempt to auto discover the API server and connect + # to it. Uncomment only if the default does not work. + # - --apiserver-host=http://my-address:port + - --token-ttl={{ dashboard_token_ttl }} + volumeMounts: + - name: kubernetes-dashboard-certs + mountPath: /certs + # Create on-disk volume to store exec logs + - mountPath: /tmp + name: tmp-volume + livenessProbe: + httpGet: + scheme: HTTPS + path: / + port: 8443 + initialDelaySeconds: 30 + timeoutSeconds: 30 + securityContext: + allowPrivilegeEscalation: false + readOnlyRootFilesystem: true + runAsUser: 1001 + runAsGroup: 2001 + volumes: + - name: kubernetes-dashboard-certs + secret: + secretName: {{ dashboard_certs_secret_name }} + - name: tmp-volume + emptyDir: {} + serviceAccountName: kubernetes-dashboard +{% if dashboard_master_toleration %} + tolerations: + - key: node-role.kubernetes.io/master + effect: NoSchedule + - key: node-role.kubernetes.io/control-plane + effect: NoSchedule +{% endif %} + +--- +# ------------------- Dashboard Service ------------------- # + +kind: Service +apiVersion: v1 +metadata: + labels: + k8s-app: kubernetes-dashboard + name: kubernetes-dashboard + namespace: {{ dashboard_namespace }} +spec: + ports: + - port: 443 + targetPort: 8443 + selector: + k8s-app: kubernetes-dashboard + +--- +# ------------------- Metrics Scrapper Service Account ------------------- # + +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: + k8s-app: kubernetes-dashboard + name: kubernetes-dashboard +rules: + # Allow Metrics Scraper to get metrics from the Metrics server + - apiGroups: ["metrics.k8s.io"] + resources: ["pods", "nodes"] + verbs: ["get", "list", "watch"] + +--- + +# ------------------- Metrics Scrapper Service ------------------- # +kind: Service +apiVersion: v1 +metadata: + labels: + k8s-app: kubernetes-metrics-scraper + name: dashboard-metrics-scraper + namespace: {{ dashboard_namespace }} +spec: + ports: + - port: 8000 + targetPort: 8000 + selector: + k8s-app: kubernetes-metrics-scraper + +--- + +# ------------------- Metrics Scrapper Deployment ------------------- # +kind: Deployment +apiVersion: apps/v1 +metadata: + labels: + k8s-app: kubernetes-metrics-scraper + name: kubernetes-metrics-scraper + namespace: {{ dashboard_namespace }} +spec: + replicas: 1 + revisionHistoryLimit: 10 + selector: + matchLabels: + k8s-app: kubernetes-metrics-scraper + template: + metadata: + labels: + k8s-app: kubernetes-metrics-scraper + spec: + securityContext: + seccompProfile: + type: RuntimeDefault + priorityClassName: system-cluster-critical + containers: + - name: kubernetes-metrics-scraper + image: {{ dashboard_metrics_scraper_repo }}:{{ dashboard_metrics_scraper_tag }} + ports: + - containerPort: 8000 + protocol: TCP + livenessProbe: + httpGet: + scheme: HTTP + path: / + port: 8000 + initialDelaySeconds: 30 + timeoutSeconds: 30 + securityContext: + allowPrivilegeEscalation: false + readOnlyRootFilesystem: true + runAsUser: 1001 + runAsGroup: 2001 + volumeMounts: + - mountPath: /tmp + name: tmp-volume + serviceAccountName: kubernetes-dashboard + volumes: + - name: tmp-volume + emptyDir: {} +{% if dashboard_master_toleration %} + tolerations: + - key: node-role.kubernetes.io/master + effect: NoSchedule + - key: node-role.kubernetes.io/control-plane + effect: NoSchedule +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler-clusterrole.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler-clusterrole.yml.j2 new file mode 100644 index 0000000..ef642ce --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler-clusterrole.yml.j2 @@ -0,0 +1,34 @@ +--- +# Copyright 2016 The Kubernetes Authors. All rights reserved +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: system:dns-autoscaler + labels: + addonmanager.kubernetes.io/mode: Reconcile +rules: + - apiGroups: [""] + resources: ["nodes"] + verbs: ["list", "watch"] + - apiGroups: [""] + resources: ["replicationcontrollers/scale"] + verbs: ["get", "update"] + - apiGroups: ["extensions", "apps"] + resources: ["deployments/scale", "replicasets/scale"] + verbs: ["get", "update"] + - apiGroups: [""] + resources: ["configmaps"] + verbs: ["get", "create"] diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler-clusterrolebinding.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler-clusterrolebinding.yml.j2 new file mode 100644 index 0000000..da1a0a9 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler-clusterrolebinding.yml.j2 @@ -0,0 +1,29 @@ +--- +# Copyright 2016 The Kubernetes Authors. All rights reserved +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: system:dns-autoscaler + labels: + addonmanager.kubernetes.io/mode: Reconcile +subjects: + - kind: ServiceAccount + name: dns-autoscaler + namespace: kube-system +roleRef: + kind: ClusterRole + name: system:dns-autoscaler + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler-sa.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler-sa.yml.j2 new file mode 100644 index 0000000..3ce9b51 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler-sa.yml.j2 @@ -0,0 +1,22 @@ +--- +# Copyright 2016 The Kubernetes Authors. All rights reserved +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +kind: ServiceAccount +apiVersion: v1 +metadata: + name: dns-autoscaler + namespace: kube-system + labels: + addonmanager.kubernetes.io/mode: Reconcile diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler.yml.j2 new file mode 100644 index 0000000..c085405 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/dns-autoscaler.yml.j2 @@ -0,0 +1,88 @@ +--- +# Copyright 2016 The Kubernetes Authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +apiVersion: apps/v1 +kind: Deployment +metadata: + name: dns-autoscaler{{ coredns_ordinal_suffix }} + namespace: kube-system + labels: + k8s-app: dns-autoscaler{{ coredns_ordinal_suffix }} + addonmanager.kubernetes.io/mode: Reconcile +spec: + selector: + matchLabels: + k8s-app: dns-autoscaler{{ coredns_ordinal_suffix }} + template: + metadata: + labels: + k8s-app: dns-autoscaler{{ coredns_ordinal_suffix }} + annotations: + spec: + nodeSelector: + {{ dns_autoscaler_deployment_nodeselector }} + priorityClassName: system-cluster-critical + securityContext: + seccompProfile: + type: RuntimeDefault + supplementalGroups: [ 65534 ] + fsGroup: 65534 + nodeSelector: + kubernetes.io/os: linux + tolerations: + - effect: NoSchedule + key: node-role.kubernetes.io/master + - effect: NoSchedule + key: node-role.kubernetes.io/control-plane +{% if dns_autoscaler_extra_tolerations is defined %} + {{ dns_autoscaler_extra_tolerations | list | to_nice_yaml(indent=2) | indent(8) }} +{% endif %} + affinity: + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - topologyKey: "kubernetes.io/hostname" + labelSelector: + matchLabels: + k8s-app: dns-autoscaler{{ coredns_ordinal_suffix }} + nodeAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - weight: 100 + preference: + matchExpressions: + - key: node-role.kubernetes.io/control-plane + operator: In + values: + - "" + containers: + - name: autoscaler + image: "{{ dnsautoscaler_image_repo }}:{{ dnsautoscaler_image_tag }}" + resources: + requests: + cpu: {{ dns_autoscaler_cpu_requests }} + memory: {{ dns_autoscaler_memory_requests }} + readinessProbe: + httpGet: + path: /healthz + port: 8080 + scheme: HTTP + command: + - /cluster-proportional-autoscaler + - --namespace=kube-system + - --default-params={"linear":{"preventSinglePointFailure":{{ dns_prevent_single_point_failure }},"coresPerReplica":{{ dns_cores_per_replica }},"nodesPerReplica":{{ dns_nodes_per_replica }},"min":{{ dns_min_replicas }}}} + - --logtostderr=true + - --v=2 + - --configmap=dns-autoscaler{{ coredns_ordinal_suffix }} + - --target=Deployment/coredns{{ coredns_ordinal_suffix }} + serviceAccountName: dns-autoscaler diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/etcd_metrics-endpoints.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/etcd_metrics-endpoints.yml.j2 new file mode 100644 index 0000000..18f515d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/etcd_metrics-endpoints.yml.j2 @@ -0,0 +1,20 @@ +apiVersion: v1 +kind: Endpoints +metadata: + name: etcd-metrics + namespace: kube-system + labels: + k8s-app: etcd + app.kubernetes.io/managed-by: Kubespray +subsets: +{% for etcd_metrics_address, etcd_host in etcd_metrics_addresses.split(',') | zip(etcd_hosts) %} + - addresses: + - ip: {{ etcd_metrics_address | urlsplit('hostname') }} + targetRef: + kind: Node + name: {{ etcd_host }} + ports: + - name: http-metrics + port: {{ etcd_metrics_address | urlsplit('port') }} + protocol: TCP +{% endfor %} diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/etcd_metrics-service.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/etcd_metrics-service.yml.j2 new file mode 100644 index 0000000..5bd9254 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/etcd_metrics-service.yml.j2 @@ -0,0 +1,13 @@ +apiVersion: v1 +kind: Service +metadata: + name: etcd-metrics + namespace: kube-system + labels: + {{ etcd_metrics_service_labels | to_yaml(indent=2, width=1337) | indent(width=4) }} +spec: + ports: + - name: http-metrics + protocol: TCP + port: {{ etcd_metrics_port }} + # targetPort: diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-ds.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-ds.yml.j2 new file mode 100644 index 0000000..40dd199 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-ds.yml.j2 @@ -0,0 +1,56 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + labels: + app: netchecker-agent + name: netchecker-agent + namespace: {{ netcheck_namespace }} +spec: + selector: + matchLabels: + app: netchecker-agent + template: + metadata: + name: netchecker-agent + labels: + app: netchecker-agent + spec: + priorityClassName: {% if netcheck_namespace == 'kube-system' %}system-node-critical{% else %}k8s-cluster-critical{% endif %}{{ '' }} + tolerations: + - effect: NoSchedule + operator: Exists + nodeSelector: + kubernetes.io/os: linux + containers: + - name: netchecker-agent + image: "{{ netcheck_agent_image_repo }}:{{ netcheck_agent_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + env: + - name: MY_POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: MY_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + args: + - "-v=5" + - "-alsologtostderr=true" + - "-serverendpoint=netchecker-service:8081" + - "-reportinterval={{ agent_report_interval }}" + resources: + limits: + cpu: {{ netchecker_agent_cpu_limit }} + memory: {{ netchecker_agent_memory_limit }} + requests: + cpu: {{ netchecker_agent_cpu_requests }} + memory: {{ netchecker_agent_memory_requests }} + securityContext: + runAsUser: {{ netchecker_agent_user | default('0') }} + runAsGroup: {{ netchecker_agent_group | default('0') }} + serviceAccountName: netchecker-agent + updateStrategy: + rollingUpdate: + maxUnavailable: 100% + type: RollingUpdate diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-clusterrole.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-clusterrole.yml.j2 new file mode 100644 index 0000000..0e23150 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-clusterrole.yml.j2 @@ -0,0 +1,14 @@ +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: psp:netchecker-agent-hostnet + namespace: {{ netcheck_namespace }} +rules: + - apiGroups: + - policy + resourceNames: + - netchecker-agent-hostnet + resources: + - podsecuritypolicies + verbs: + - use diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-clusterrolebinding.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-clusterrolebinding.yml.j2 new file mode 100644 index 0000000..cf44515 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-clusterrolebinding.yml.j2 @@ -0,0 +1,13 @@ +kind: RoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: psp:netchecker-agent-hostnet + namespace: {{ netcheck_namespace }} +subjects: + - kind: ServiceAccount + name: netchecker-agent + namespace: {{ netcheck_namespace }} +roleRef: + kind: ClusterRole + name: psp:netchecker-agent-hostnet + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-ds.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-ds.yml.j2 new file mode 100644 index 0000000..50e2793 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-ds.yml.j2 @@ -0,0 +1,58 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + labels: + app: netchecker-agent-hostnet + name: netchecker-agent-hostnet + namespace: {{ netcheck_namespace }} +spec: + selector: + matchLabels: + app: netchecker-agent-hostnet + template: + metadata: + name: netchecker-agent-hostnet + labels: + app: netchecker-agent-hostnet + spec: + hostNetwork: true + dnsPolicy: ClusterFirstWithHostNet + nodeSelector: + kubernetes.io/os: linux + priorityClassName: {% if netcheck_namespace == 'kube-system' %}system-node-critical{% else %}k8s-cluster-critical{% endif %}{{ '' }} + tolerations: + - effect: NoSchedule + operator: Exists + containers: + - name: netchecker-agent + image: "{{ netcheck_agent_image_repo }}:{{ netcheck_agent_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + env: + - name: MY_POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: MY_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + args: + - "-v=5" + - "-alsologtostderr=true" + - "-serverendpoint=netchecker-service:8081" + - "-reportinterval={{ agent_report_interval }}" + resources: + limits: + cpu: {{ netchecker_agent_cpu_limit }} + memory: {{ netchecker_agent_memory_limit }} + requests: + cpu: {{ netchecker_agent_cpu_requests }} + memory: {{ netchecker_agent_memory_requests }} + securityContext: + runAsUser: {{ netchecker_agent_user | default('0') }} + runAsGroup: {{ netchecker_agent_group | default('0') }} + serviceAccountName: netchecker-agent + updateStrategy: + rollingUpdate: + maxUnavailable: 100% + type: RollingUpdate diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-psp.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-psp.yml.j2 new file mode 100644 index 0000000..21b397d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-hostnet-psp.yml.j2 @@ -0,0 +1,44 @@ +--- +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: netchecker-agent-hostnet + annotations: + seccomp.security.alpha.kubernetes.io/defaultProfileName: 'runtime/default' + seccomp.security.alpha.kubernetes.io/allowedProfileNames: 'runtime/default' +{% if apparmor_enabled %} + apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + apparmor.security.beta.kubernetes.io/allowedProfileNames: 'runtime/default' +{% endif %} + labels: + addonmanager.kubernetes.io/mode: Reconcile +spec: + privileged: false + allowPrivilegeEscalation: false + requiredDropCapabilities: + - ALL + volumes: + - 'configMap' + - 'emptyDir' + - 'projected' + - 'secret' + - 'downwardAPI' + - 'persistentVolumeClaim' + hostNetwork: true + hostIPC: false + hostPID: false + runAsUser: + rule: 'MustRunAsNonRoot' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + readOnlyRootFilesystem: false diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-sa.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-sa.yml.j2 new file mode 100644 index 0000000..c544043 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-agent-sa.yml.j2 @@ -0,0 +1,5 @@ +apiVersion: v1 +kind: ServiceAccount +metadata: + name: netchecker-agent + namespace: {{ netcheck_namespace }} diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-ns.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-ns.yml.j2 new file mode 100644 index 0000000..3dd87aa --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-ns.yml.j2 @@ -0,0 +1,6 @@ +apiVersion: v1 +kind: Namespace +metadata: + name: "{{ netcheck_namespace }}" + labels: + name: "{{ netcheck_namespace }}" diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-clusterrole.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-clusterrole.yml.j2 new file mode 100644 index 0000000..290dec3 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-clusterrole.yml.j2 @@ -0,0 +1,9 @@ +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: netchecker-server + namespace: {{ netcheck_namespace }} +rules: + - apiGroups: [""] + resources: ["pods"] + verbs: ["list", "get"] diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-clusterrolebinding.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-clusterrolebinding.yml.j2 new file mode 100644 index 0000000..55301b7 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-clusterrolebinding.yml.j2 @@ -0,0 +1,13 @@ +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: netchecker-server + namespace: {{ netcheck_namespace }} +subjects: + - kind: ServiceAccount + name: netchecker-server + namespace: {{ netcheck_namespace }} +roleRef: + kind: ClusterRole + name: netchecker-server + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-deployment.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-deployment.yml.j2 new file mode 100644 index 0000000..02fd6b6 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-deployment.yml.j2 @@ -0,0 +1,83 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: netchecker-server + namespace: {{ netcheck_namespace }} + labels: + app: netchecker-server +spec: + replicas: 1 + selector: + matchLabels: + app: netchecker-server + template: + metadata: + name: netchecker-server + labels: + app: netchecker-server + spec: + priorityClassName: {% if netcheck_namespace == 'kube-system' %}system-cluster-critical{% else %}k8s-cluster-critical{% endif %}{{ '' }} + volumes: + - name: etcd-data + emptyDir: {} + containers: + - name: netchecker-server + image: "{{ netcheck_server_image_repo }}:{{ netcheck_server_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + resources: + limits: + cpu: {{ netchecker_server_cpu_limit }} + memory: {{ netchecker_server_memory_limit }} + requests: + cpu: {{ netchecker_server_cpu_requests }} + memory: {{ netchecker_server_memory_requests }} + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: ['ALL'] + runAsUser: {{ netchecker_server_user | default('0') }} + runAsGroup: {{ netchecker_server_group | default('0') }} + runAsNonRoot: true + seccompProfile: + type: RuntimeDefault + ports: + - containerPort: 8081 + args: + - -v=5 + - -logtostderr + - -kubeproxyinit=false + - -endpoint=0.0.0.0:8081 + - -etcd-endpoints=http://127.0.0.1:2379 + - name: etcd + image: "{{ etcd_image_repo }}:{{ netcheck_etcd_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + command: + - etcd + - --listen-client-urls=http://127.0.0.1:2379 + - --advertise-client-urls=http://127.0.0.1:2379 + - --data-dir=/var/lib/etcd + - --enable-v2 + - --force-new-cluster + volumeMounts: + - mountPath: /var/lib/etcd + name: etcd-data + resources: + limits: + cpu: {{ netchecker_etcd_cpu_limit }} + memory: {{ netchecker_etcd_memory_limit }} + requests: + cpu: {{ netchecker_etcd_cpu_requests }} + memory: {{ netchecker_etcd_memory_requests }} + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: ['ALL'] + runAsUser: {{ netchecker_server_user | default('0') }} + runAsGroup: {{ netchecker_server_group | default('0') }} + runAsNonRoot: true + seccompProfile: + type: RuntimeDefault + tolerations: + - effect: NoSchedule + operator: Exists + serviceAccountName: netchecker-server diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-sa.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-sa.yml.j2 new file mode 100644 index 0000000..e3ec07f --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-sa.yml.j2 @@ -0,0 +1,5 @@ +apiVersion: v1 +kind: ServiceAccount +metadata: + name: netchecker-server + namespace: {{ netcheck_namespace }} diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-svc.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-svc.yml.j2 new file mode 100644 index 0000000..dc38946 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/netchecker-server-svc.yml.j2 @@ -0,0 +1,15 @@ +apiVersion: v1 +kind: Service +metadata: + name: netchecker-service + namespace: {{ netcheck_namespace }} +spec: + selector: + app: netchecker-server + ports: + - + protocol: TCP + port: 8081 + targetPort: 8081 + nodePort: {{ netchecker_port }} + type: NodePort diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-config.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-config.yml.j2 new file mode 100644 index 0000000..b15ea89 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-config.yml.j2 @@ -0,0 +1,182 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: nodelocaldns + namespace: kube-system + labels: + addonmanager.kubernetes.io/mode: EnsureExists + +data: + Corefile: | +{% if nodelocaldns_external_zones is defined and nodelocaldns_external_zones | length > 0 %} +{% for block in nodelocaldns_external_zones %} + {{ block['zones'] | join(' ') }} { + errors + cache {{ block['cache'] | default(30) }} + reload +{% if block['rewrite'] is defined and block['rewrite'] | length > 0 %} +{% for rewrite_match in block['rewrite'] %} + rewrite {{ rewrite_match }} +{% endfor %} +{% endif %} + loop + bind {{ nodelocaldns_ip }} + forward . {{ block['nameservers'] | join(' ') }} + prometheus {% if nodelocaldns_bind_metrics_host_ip %}{$MY_HOST_IP}{% endif %}:{{ nodelocaldns_prometheus_port }} + log +{% if dns_etchosts | default(None) %} + hosts /etc/coredns/hosts { + fallthrough + } +{% endif %} + } +{% endfor %} +{% endif %} + {{ dns_domain }}:53 { + errors + cache { + success 9984 30 + denial 9984 5 + } + reload + loop + bind {{ nodelocaldns_ip }} + forward . {{ forwardTarget }} { + force_tcp + } + prometheus {% if nodelocaldns_bind_metrics_host_ip %}{$MY_HOST_IP}{% endif %}:{{ nodelocaldns_prometheus_port }} + health {{ nodelocaldns_ip }}:{{ nodelocaldns_health_port }} +{% if dns_etchosts | default(None) %} + hosts /etc/coredns/hosts { + fallthrough + } +{% endif %} + } + in-addr.arpa:53 { + errors + cache 30 + reload + loop + bind {{ nodelocaldns_ip }} + forward . {{ forwardTarget }} { + force_tcp + } + prometheus {% if nodelocaldns_bind_metrics_host_ip %}{$MY_HOST_IP}{% endif %}:{{ nodelocaldns_prometheus_port }} + } + ip6.arpa:53 { + errors + cache 30 + reload + loop + bind {{ nodelocaldns_ip }} + forward . {{ forwardTarget }} { + force_tcp + } + prometheus {% if nodelocaldns_bind_metrics_host_ip %}{$MY_HOST_IP}{% endif %}:{{ nodelocaldns_prometheus_port }} + } + .:53 { + errors + cache 30 + reload + loop + bind {{ nodelocaldns_ip }} + forward . {{ upstreamForwardTarget }}{% if dns_upstream_forward_extra_opts is defined %} { +{% for optname, optvalue in dns_upstream_forward_extra_opts.items() %} + {{ optname }} {{ optvalue }} +{% endfor %} + }{% endif %} + + prometheus {% if nodelocaldns_bind_metrics_host_ip %}{$MY_HOST_IP}{% endif %}:{{ nodelocaldns_prometheus_port }} +{% if dns_etchosts | default(None) %} + hosts /etc/coredns/hosts { + fallthrough + } +{% endif %} + } +{% if enable_nodelocaldns_secondary %} + Corefile-second: | +{% if nodelocaldns_external_zones is defined and nodelocaldns_external_zones | length > 0 %} +{% for block in nodelocaldns_external_zones %} + {{ block['zones'] | join(' ') }} { + errors + cache {{ block['cache'] | default(30) }} + reload + loop + bind {{ nodelocaldns_ip }} + forward . {{ block['nameservers'] | join(' ') }} + prometheus {% if nodelocaldns_bind_metrics_host_ip %}{$MY_HOST_IP}{% endif %}:{{ nodelocaldns_secondary_prometheus_port }} + log +{% if dns_etchosts | default(None) %} + hosts /etc/coredns/hosts { + fallthrough + } +{% endif %} + } +{% endfor %} +{% endif %} + {{ dns_domain }}:53 { + errors + cache { + success 9984 30 + denial 9984 5 + } + reload + loop + bind {{ nodelocaldns_ip }} + forward . {{ forwardTarget }} { + force_tcp + } + prometheus {% if nodelocaldns_bind_metrics_host_ip %}{$MY_HOST_IP}{% endif %}:{{ nodelocaldns_secondary_prometheus_port }} + health {{ nodelocaldns_ip }}:{{ nodelocaldns_second_health_port }} +{% if dns_etchosts | default(None) %} + hosts /etc/coredns/hosts { + fallthrough + } +{% endif %} + } + in-addr.arpa:53 { + errors + cache 30 + reload + loop + bind {{ nodelocaldns_ip }} + forward . {{ forwardTarget }} { + force_tcp + } + prometheus {% if nodelocaldns_bind_metrics_host_ip %}{$MY_HOST_IP}{% endif %}:{{ nodelocaldns_secondary_prometheus_port }} + } + ip6.arpa:53 { + errors + cache 30 + reload + loop + bind {{ nodelocaldns_ip }} + forward . {{ forwardTarget }} { + force_tcp + } + prometheus {% if nodelocaldns_bind_metrics_host_ip %}{$MY_HOST_IP}{% endif %}:{{ nodelocaldns_secondary_prometheus_port }} + } + .:53 { + errors + cache 30 + reload + loop + bind {{ nodelocaldns_ip }} + forward . {{ upstreamForwardTarget }}{% if dns_upstream_forward_extra_opts is defined %} { +{% for optname, optvalue in dns_upstream_forward_extra_opts.items() %} + {{ optname }} {{ optvalue }} +{% endfor %} + }{% endif %} + + prometheus {% if nodelocaldns_bind_metrics_host_ip %}{$MY_HOST_IP}{% endif %}:{{ nodelocaldns_secondary_prometheus_port }} +{% if dns_etchosts | default(None) %} + hosts /etc/coredns/hosts { + fallthrough + } +{% endif %} + } +{% endif %} +{% if dns_etchosts | default(None) %} + hosts: | + {{ dns_etchosts | indent(width=4, first=False) }} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-daemonset.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-daemonset.yml.j2 new file mode 100644 index 0000000..9ca15d7 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-daemonset.yml.j2 @@ -0,0 +1,115 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: nodelocaldns + namespace: kube-system + labels: + k8s-app: kube-dns + addonmanager.kubernetes.io/mode: Reconcile +spec: + selector: + matchLabels: + k8s-app: node-local-dns + template: + metadata: + labels: + k8s-app: node-local-dns + annotations: + prometheus.io/scrape: 'true' + prometheus.io/port: '{{ nodelocaldns_prometheus_port }}' + spec: + nodeSelector: + {{ nodelocaldns_ds_nodeselector }} + priorityClassName: system-cluster-critical + serviceAccountName: nodelocaldns + hostNetwork: true + dnsPolicy: Default # Don't use cluster DNS. + tolerations: + - effect: NoSchedule + operator: "Exists" + - effect: NoExecute + operator: "Exists" + containers: + - name: node-cache + image: "{{ nodelocaldns_image_repo }}:{{ nodelocaldns_image_tag }}" + resources: + limits: + memory: {{ nodelocaldns_memory_limit }} + requests: + cpu: {{ nodelocaldns_cpu_requests }} + memory: {{ nodelocaldns_memory_requests }} + args: + - -localip + - {{ nodelocaldns_ip }} + - -conf + - /etc/coredns/Corefile + - -upstreamsvc + - coredns +{% if enable_nodelocaldns_secondary %} + - -skipteardown +{% else %} + ports: + - containerPort: 53 + name: dns + protocol: UDP + - containerPort: 53 + name: dns-tcp + protocol: TCP + - containerPort: 9253 + name: metrics + protocol: TCP +{% endif %} + securityContext: + privileged: true +{% if nodelocaldns_bind_metrics_host_ip %} + env: + - name: MY_HOST_IP + valueFrom: + fieldRef: + fieldPath: status.hostIP +{% endif %} + livenessProbe: + httpGet: + host: {{ nodelocaldns_ip }} + path: /health + port: {{ nodelocaldns_health_port }} + scheme: HTTP + timeoutSeconds: 5 + successThreshold: 1 + failureThreshold: 10 + readinessProbe: + httpGet: + host: {{ nodelocaldns_ip }} + path: /health + port: {{ nodelocaldns_health_port }} + scheme: HTTP + timeoutSeconds: 5 + successThreshold: 1 + failureThreshold: 10 + volumeMounts: + - name: config-volume + mountPath: /etc/coredns + - name: xtables-lock + mountPath: /run/xtables.lock + volumes: + - name: config-volume + configMap: + name: nodelocaldns + items: + - key: Corefile + path: Corefile +{% if dns_etchosts | default(None) %} + - key: hosts + path: hosts +{% endif %} + - name: xtables-lock + hostPath: + path: /run/xtables.lock + type: FileOrCreate + # Minimize downtime during a rolling upgrade or deletion; tell Kubernetes to do a "force + # deletion": https://kubernetes.io/docs/concepts/workloads/pods/pod/#termination-of-pods. + terminationGracePeriodSeconds: 0 + updateStrategy: + rollingUpdate: + maxUnavailable: {{ serial | default('20%') }} + type: RollingUpdate diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-sa.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-sa.yml.j2 new file mode 100644 index 0000000..bd962d8 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-sa.yml.j2 @@ -0,0 +1,7 @@ +apiVersion: v1 +kind: ServiceAccount +metadata: + name: nodelocaldns + namespace: kube-system + labels: + addonmanager.kubernetes.io/mode: Reconcile diff --git a/kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-second-daemonset.yml.j2 b/kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-second-daemonset.yml.j2 new file mode 100644 index 0000000..df9405e --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ansible/templates/nodelocaldns-second-daemonset.yml.j2 @@ -0,0 +1,103 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: nodelocaldns-second + namespace: kube-system + labels: + k8s-app: kube-dns + addonmanager.kubernetes.io/mode: Reconcile +spec: + selector: + matchLabels: + k8s-app: node-local-dns-second + template: + metadata: + labels: + k8s-app: node-local-dns-second + annotations: + prometheus.io/scrape: 'true' + prometheus.io/port: '{{ nodelocaldns_secondary_prometheus_port }}' + spec: + nodeSelector: + {{ nodelocaldns_ds_nodeselector }} + priorityClassName: system-cluster-critical + serviceAccountName: nodelocaldns + hostNetwork: true + dnsPolicy: Default # Don't use cluster DNS. + tolerations: + - effect: NoSchedule + operator: "Exists" + - effect: NoExecute + operator: "Exists" + containers: + - name: node-cache + image: "{{ nodelocaldns_image_repo }}:{{ nodelocaldns_image_tag }}" + resources: + limits: + memory: {{ nodelocaldns_memory_limit }} + requests: + cpu: {{ nodelocaldns_cpu_requests }} + memory: {{ nodelocaldns_memory_requests }} + args: [ "-localip", "{{ nodelocaldns_ip }}", "-conf", "/etc/coredns/Corefile", "-upstreamsvc", "coredns", "-skipteardown" ] + securityContext: + privileged: true +{% if nodelocaldns_bind_metrics_host_ip %} + env: + - name: MY_HOST_IP + valueFrom: + fieldRef: + fieldPath: status.hostIP +{% endif %} + livenessProbe: + httpGet: + host: {{ nodelocaldns_ip }} + path: /health + port: {{ nodelocaldns_health_port }} + scheme: HTTP + timeoutSeconds: 5 + successThreshold: 1 + failureThreshold: 10 + readinessProbe: + httpGet: + host: {{ nodelocaldns_ip }} + path: /health + port: {{ nodelocaldns_health_port }} + scheme: HTTP + timeoutSeconds: 5 + successThreshold: 1 + failureThreshold: 10 + volumeMounts: + - name: config-volume + mountPath: /etc/coredns + - name: xtables-lock + mountPath: /run/xtables.lock + lifecycle: + preStop: + exec: + command: + - sh + - -c + - sleep {{ nodelocaldns_secondary_skew_seconds }} && kill -9 1 + volumes: + - name: config-volume + configMap: + name: nodelocaldns + items: + - key: Corefile-second + path: Corefile +{% if dns_etchosts | default(None) %} + - key: hosts + path: hosts +{% endif %} + - name: xtables-lock + hostPath: + path: /run/xtables.lock + type: FileOrCreate + # Implement a time skew between the main nodelocaldns and this secondary. + # Since the two nodelocaldns instances share the :53 port, we want to keep + # at least one running at any time enven if the manifests are replaced simultaneously + terminationGracePeriodSeconds: {{ nodelocaldns_secondary_skew_seconds }} + updateStrategy: + rollingUpdate: + maxUnavailable: {{ serial | default('20%') }} + type: RollingUpdate diff --git a/kubespray/roles/kubernetes-apps/argocd/defaults/main.yml b/kubespray/roles/kubernetes-apps/argocd/defaults/main.yml new file mode 100644 index 0000000..e4507d4 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/argocd/defaults/main.yml @@ -0,0 +1,6 @@ +--- +argocd_enabled: false +argocd_version: v2.8.0 +argocd_namespace: argocd +# argocd_admin_password: +argocd_install_url: "https://raw.githubusercontent.com/argoproj/argo-cd/{{ argocd_version }}/manifests/install.yaml" diff --git a/kubespray/roles/kubernetes-apps/argocd/tasks/main.yml b/kubespray/roles/kubernetes-apps/argocd/tasks/main.yml new file mode 100644 index 0000000..e11f097 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/argocd/tasks/main.yml @@ -0,0 +1,107 @@ +--- +- name: Kubernetes Apps | Download yq + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.yq) }}" + +- name: Kubernetes Apps | Copy yq binary from download dir + ansible.posix.synchronize: + src: "{{ downloads.yq.dest }}" + dest: "{{ bin_dir }}/yq" + compress: no + perms: yes + owner: no + group: no + delegate_to: "{{ inventory_hostname }}" + +- name: Kubernetes Apps | Set ArgoCD template list + set_fact: + argocd_templates: + - name: namespace + file: argocd-namespace.yml + - name: install + file: argocd-install.yml + namespace: "{{ argocd_namespace }}" + url: "{{ argocd_install_url }}" + when: + - "inventory_hostname == groups['kube_control_plane'][0]" + +- name: Kubernetes Apps | Download ArgoCD remote manifests + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download_argocd: + enabled: "{{ argocd_enabled }}" + file: true + dest: "{{ local_release_dir }}/{{ item.file }}" + url: "{{ item.url }}" + unarchive: false + owner: "root" + mode: 0644 + sha256: "" + download: "{{ download_defaults | combine(download_argocd) }}" + with_items: "{{ argocd_templates | selectattr('url', 'defined') | list }}" + loop_control: + label: "{{ item.file }}" + when: + - "inventory_hostname == groups['kube_control_plane'][0]" + +- name: Kubernetes Apps | Copy ArgoCD remote manifests from download dir + ansible.posix.synchronize: + src: "{{ local_release_dir }}/{{ item.file }}" + dest: "{{ kube_config_dir }}/{{ item.file }}" + compress: no + perms: yes + owner: no + group: no + delegate_to: "{{ inventory_hostname }}" + with_items: "{{ argocd_templates | selectattr('url', 'defined') | list }}" + when: + - "inventory_hostname == groups['kube_control_plane'][0]" + +- name: Kubernetes Apps | Set ArgoCD namespace for remote manifests + become: yes + command: | + {{ bin_dir }}/yq eval-all -i '.metadata.namespace="{{ argocd_namespace }}"' {{ kube_config_dir }}/{{ item.file }} + with_items: "{{ argocd_templates | selectattr('url', 'defined') | list }}" + loop_control: + label: "{{ item.file }}" + when: + - "inventory_hostname == groups['kube_control_plane'][0]" + +- name: Kubernetes Apps | Create ArgoCD manifests from templates + become: yes + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: "{{ argocd_templates | selectattr('url', 'undefined') | list }}" + loop_control: + label: "{{ item.file }}" + when: + - "inventory_hostname == groups['kube_control_plane'][0]" + +- name: Kubernetes Apps | Install ArgoCD + become: yes + kube: + name: ArgoCD + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.file }}" + state: latest + with_items: "{{ argocd_templates }}" + when: + - "inventory_hostname == groups['kube_control_plane'][0]" + +# https://github.com/argoproj/argo-cd/blob/master/docs/faq.md#i-forgot-the-admin-password-how-do-i-reset-it +- name: Kubernetes Apps | Set ArgoCD custom admin password + become: yes + shell: | + {{ bin_dir }}/kubectl --kubeconfig /etc/kubernetes/admin.conf -n {{ argocd_namespace }} patch secret argocd-secret -p \ + '{ + "stringData": { + "admin.password": "{{ argocd_admin_password | password_hash('bcrypt') }}", + "admin.passwordMtime": "'$(date +%FT%T%Z)'" + } + }' + when: + - argocd_admin_password is defined + - "inventory_hostname == groups['kube_control_plane'][0]" diff --git a/kubespray/roles/kubernetes-apps/argocd/templates/argocd-namespace.yml.j2 b/kubespray/roles/kubernetes-apps/argocd/templates/argocd-namespace.yml.j2 new file mode 100644 index 0000000..5a3d40f --- /dev/null +++ b/kubespray/roles/kubernetes-apps/argocd/templates/argocd-namespace.yml.j2 @@ -0,0 +1,7 @@ +--- +apiVersion: v1 +kind: Namespace +metadata: + name: {{ argocd_namespace }} + labels: + app: argocd diff --git a/kubespray/roles/kubernetes-apps/cloud_controller/oci/defaults/main.yml b/kubespray/roles/kubernetes-apps/cloud_controller/oci/defaults/main.yml new file mode 100644 index 0000000..9d7ddf0 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/cloud_controller/oci/defaults/main.yml @@ -0,0 +1,6 @@ +--- + +oci_security_list_management: All +oci_use_instance_principals: false +oci_cloud_controller_version: 0.7.0 +oci_cloud_controller_pull_source: iad.ocir.io/oracle/cloud-provider-oci diff --git a/kubespray/roles/kubernetes-apps/cloud_controller/oci/tasks/credentials-check.yml b/kubespray/roles/kubernetes-apps/cloud_controller/oci/tasks/credentials-check.yml new file mode 100644 index 0000000..9eb8794 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/cloud_controller/oci/tasks/credentials-check.yml @@ -0,0 +1,67 @@ +--- + +- name: "OCI Cloud Controller | Credentials Check | oci_private_key" + fail: + msg: "oci_private_key is missing" + when: + - not oci_use_instance_principals + - oci_private_key is not defined or not oci_private_key + +- name: "OCI Cloud Controller | Credentials Check | oci_region_id" + fail: + msg: "oci_region_id is missing" + when: + - not oci_use_instance_principals + - oci_region_id is not defined or not oci_region_id + +- name: "OCI Cloud Controller | Credentials Check | oci_tenancy_id" + fail: + msg: "oci_tenancy_id is missing" + when: + - not oci_use_instance_principals + - oci_tenancy_id is not defined or not oci_tenancy_id + +- name: "OCI Cloud Controller | Credentials Check | oci_user_id" + fail: + msg: "oci_user_id is missing" + when: + - not oci_use_instance_principals + - oci_user_id is not defined or not oci_user_id + +- name: "OCI Cloud Controller | Credentials Check | oci_user_fingerprint" + fail: + msg: "oci_user_fingerprint is missing" + when: + - not oci_use_instance_principals + - oci_user_fingerprint is not defined or not oci_user_fingerprint + +- name: "OCI Cloud Controller | Credentials Check | oci_compartment_id" + fail: + msg: "oci_compartment_id is missing. This is the compartment in which the cluster resides" + when: + - oci_compartment_id is not defined or not oci_compartment_id + +- name: "OCI Cloud Controller | Credentials Check | oci_vnc_id" + fail: + msg: "oci_vnc_id is missing. This is the Virtual Cloud Network in which the cluster resides" + when: + - oci_vnc_id is not defined or not oci_vnc_id + +- name: "OCI Cloud Controller | Credentials Check | oci_subnet1_id" + fail: + msg: "oci_subnet1_id is missingg. This is the first subnet to which loadbalancers will be added" + when: + - oci_subnet1_id is not defined or not oci_subnet1_id + +- name: "OCI Cloud Controller | Credentials Check | oci_subnet2_id" + fail: + msg: "oci_subnet2_id is missing. Two subnets are required for load balancer high availability" + when: + - oci_cloud_controller_version is version_compare('0.7.0', '<') + - oci_subnet2_id is not defined or not oci_subnet2_id + +- name: "OCI Cloud Controller | Credentials Check | oci_security_list_management" + fail: + msg: "oci_security_list_management is missing, or not defined correctly. Valid options are (All, Frontend, None)." + when: + - oci_security_list_management is not defined or oci_security_list_management not in ["All", "Frontend", "None"] diff --git a/kubespray/roles/kubernetes-apps/cloud_controller/oci/tasks/main.yml b/kubespray/roles/kubernetes-apps/cloud_controller/oci/tasks/main.yml new file mode 100644 index 0000000..6bfcc25 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/cloud_controller/oci/tasks/main.yml @@ -0,0 +1,35 @@ +--- + +- name: OCI Cloud Controller | Check Oracle Cloud credentials + import_tasks: credentials-check.yml + +- name: "OCI Cloud Controller | Generate Cloud Provider Configuration" + template: + src: controller-manager-config.yml.j2 + dest: "{{ kube_config_dir }}/controller-manager-config.yml" + mode: 0644 + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: "OCI Cloud Controller | Slurp Configuration" + slurp: + src: "{{ kube_config_dir }}/controller-manager-config.yml" + register: controller_manager_config + +- name: "OCI Cloud Controller | Encode Configuration" + set_fact: + controller_manager_config_base64: "{{ controller_manager_config.content }}" + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: "OCI Cloud Controller | Generate Manifests" + template: + src: oci-cloud-provider.yml.j2 + dest: "{{ kube_config_dir }}/oci-cloud-provider.yml" + mode: 0644 + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: "OCI Cloud Controller | Apply Manifests" + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/oci-cloud-provider.yml" + state: latest + when: inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/cloud_controller/oci/templates/controller-manager-config.yml.j2 b/kubespray/roles/kubernetes-apps/cloud_controller/oci/templates/controller-manager-config.yml.j2 new file mode 100644 index 0000000..d585de1 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/cloud_controller/oci/templates/controller-manager-config.yml.j2 @@ -0,0 +1,89 @@ +{% macro private_key() %}{{ oci_private_key }}{% endmacro %} + +{% if oci_use_instance_principals %} + # (https://docs.us-phoenix-1.oraclecloud.com/Content/Identity/Tasks/callingservicesfrominstances.htm). + # Ensure you have setup the following OCI policies and your kubernetes nodes are running within them + # allow dynamic-group [your dynamic group name] to read instance-family in compartment [your compartment name] + # allow dynamic-group [your dynamic group name] to use virtual-network-family in compartment [your compartment name] + # allow dynamic-group [your dynamic group name] to manage load-balancers in compartment [your compartment name] +useInstancePrincipals: true +{% else %} +useInstancePrincipals: false +{% endif %} + +auth: + +{% if oci_use_instance_principals %} + # This key is put here too for backwards compatibility + useInstancePrincipals: true +{% else %} + useInstancePrincipals: false + + region: {{ oci_region_id }} + tenancy: {{ oci_tenancy_id }} + user: {{ oci_user_id }} + key: | + {{ oci_private_key }} + + {% if oci_private_key_passphrase is defined %} + passphrase: {{ oci_private_key_passphrase }} + {% endif %} + + + fingerprint: {{ oci_user_fingerprint }} +{% endif %} + +# compartment configures Compartment within which the cluster resides. +compartment: {{ oci_compartment_id }} + +# vcn configures the Virtual Cloud Network (VCN) within which the cluster resides. +vcn: {{ oci_vnc_id }} + +loadBalancer: + # subnet1 configures one of two subnets to which load balancers will be added. + # OCI load balancers require two subnets to ensure high availability. + subnet1: {{ oci_subnet1_id }} +{% if oci_subnet2_id is defined %} + # subnet2 configures the second of two subnets to which load balancers will be + # added. OCI load balancers require two subnets to ensure high availability. + subnet2: {{ oci_subnet2_id }} +{% endif %} + # SecurityListManagementMode configures how security lists are managed by the CCM. + # "All" (default): Manage all required security list rules for load balancer services. + # "Frontend": Manage only security list rules for ingress to the load + # balancer. Requires that the user has setup a rule that + # allows inbound traffic to the appropriate ports for kube + # proxy health port, node port ranges, and health check port ranges. + # E.g. 10.82.0.0/16 30000-32000. + # "None": Disables all security list management. Requires that the + # user has setup a rule that allows inbound traffic to the + # appropriate ports for kube proxy health port, node port + # ranges, and health check port ranges. E.g. 10.82.0.0/16 30000-32000. + # Additionally requires the user to mange rules to allow + # inbound traffic to load balancers. + securityListManagementMode: {{ oci_security_list_management }} + +{% if oci_security_lists is defined and oci_security_lists | length > 0 %} + # Optional specification of which security lists to modify per subnet. This does not apply if security list management is off. + securityLists: +{% for subnet_ocid, list_ocid in oci_security_lists.items() %} + {{ subnet_ocid }}: {{ list_ocid }} +{% endfor %} +{% endif %} + +{% if oci_rate_limit is defined and oci_rate_limit | length > 0 %} +# Optional rate limit controls for accessing OCI API +rateLimiter: +{% if oci_rate_limit.rate_limit_qps_read %} + rateLimitQPSRead: {{ oci_rate_limit.rate_limit_qps_read }} +{% endif %} +{% if oci_rate_limit.rate_limit_qps_write %} + rateLimitQPSWrite: {{ oci_rate_limit.rate_limit_qps_write }} +{% endif %} +{% if oci_rate_limit.rate_limit_bucket_read %} + rateLimitBucketRead: {{ oci_rate_limit.rate_limit_bucket_read }} +{% endif %} +{% if oci_rate_limit.rate_limit_bucket_write %} + rateLimitBucketWrite: {{ oci_rate_limit.rate_limit_bucket_write }} +{% endif %} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/cloud_controller/oci/templates/oci-cloud-provider.yml.j2 b/kubespray/roles/kubernetes-apps/cloud_controller/oci/templates/oci-cloud-provider.yml.j2 new file mode 100644 index 0000000..6b45d81 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/cloud_controller/oci/templates/oci-cloud-provider.yml.j2 @@ -0,0 +1,72 @@ +apiVersion: v1 +data: + cloud-provider.yaml: {{ controller_manager_config_base64 }} +kind: Secret +metadata: + name: oci-cloud-controller-manager + namespace: kube-system +type: Opaque + +--- +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: oci-cloud-controller-manager + namespace: kube-system + labels: + k8s-app: oci-cloud-controller-manager +spec: + selector: + matchLabels: + component: oci-cloud-controller-manager + tier: control-plane + updateStrategy: + type: RollingUpdate + template: + metadata: + labels: + component: oci-cloud-controller-manager + tier: control-plane + spec: +{% if oci_cloud_controller_pull_secret is defined %} + imagePullSecrets: + - name: {{ oci_cloud_controller_pull_secret }} +{% endif %} + serviceAccountName: cloud-controller-manager + hostNetwork: true + dnsPolicy: ClusterFirstWithHostNet + nodeSelector: + node-role.kubernetes.io/control-plane: "" + tolerations: + - key: node.cloudprovider.kubernetes.io/uninitialized + value: "true" + effect: NoSchedule + - key: node-role.kubernetes.io/master + operator: Exists + effect: NoSchedule + - key: node-role.kubernetes.io/control-plane + operator: Exists + effect: NoSchedule + volumes: + - name: cfg + secret: + secretName: oci-cloud-controller-manager + - name: kubernetes + hostPath: + path: /etc/kubernetes + containers: + - name: oci-cloud-controller-manager + image: {{ oci_cloud_controller_pull_source }}:{{ oci_cloud_controller_version }} + command: ["/usr/local/bin/oci-cloud-controller-manager"] + args: + - --cloud-config=/etc/oci/cloud-provider.yaml + - --cloud-provider=oci + - --leader-elect-resource-lock=configmaps + - -v=2 + volumeMounts: + - name: cfg + mountPath: /etc/oci + readOnly: true + - name: kubernetes + mountPath: /etc/kubernetes + readOnly: true diff --git a/kubespray/roles/kubernetes-apps/cluster_roles/defaults/main.yml b/kubespray/roles/kubernetes-apps/cluster_roles/defaults/main.yml new file mode 100644 index 0000000..f26583d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/cluster_roles/defaults/main.yml @@ -0,0 +1,65 @@ +--- + +podsecuritypolicy_restricted_spec: + privileged: false + allowPrivilegeEscalation: false + requiredDropCapabilities: + - ALL + volumes: + - 'configMap' + - 'emptyDir' + - 'projected' + - 'secret' + - 'downwardAPI' + - 'persistentVolumeClaim' + hostNetwork: false + hostIPC: false + hostPID: false + runAsUser: + rule: 'MustRunAsNonRoot' + seLinux: + rule: 'RunAsAny' + runAsGroup: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + supplementalGroups: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + readOnlyRootFilesystem: false + +podsecuritypolicy_privileged_spec: + privileged: true + allowPrivilegeEscalation: true + allowedCapabilities: + - '*' + volumes: + - '*' + hostNetwork: true + hostPorts: + - min: 0 + max: 65535 + hostIPC: true + hostPID: true + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + runAsGroup: + rule: 'RunAsAny' + supplementalGroups: + rule: 'RunAsAny' + fsGroup: + rule: 'RunAsAny' + readOnlyRootFilesystem: false + # This will fail if allowed-unsafe-sysctls is not set accordingly in kubelet flags + allowedUnsafeSysctls: + - '*' diff --git a/kubespray/roles/kubernetes-apps/cluster_roles/files/k8s-cluster-critical-pc.yml b/kubespray/roles/kubernetes-apps/cluster_roles/files/k8s-cluster-critical-pc.yml new file mode 100644 index 0000000..479fb57 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/cluster_roles/files/k8s-cluster-critical-pc.yml @@ -0,0 +1,8 @@ +--- +apiVersion: scheduling.k8s.io/v1 +kind: PriorityClass +metadata: + name: k8s-cluster-critical +value: 1000000000 +globalDefault: false +description: "This priority class should only be used by the pods installed using kubespray." diff --git a/kubespray/roles/kubernetes-apps/cluster_roles/files/oci-rbac.yml b/kubespray/roles/kubernetes-apps/cluster_roles/files/oci-rbac.yml new file mode 100644 index 0000000..5e3b82b --- /dev/null +++ b/kubespray/roles/kubernetes-apps/cluster_roles/files/oci-rbac.yml @@ -0,0 +1,124 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: cloud-controller-manager + namespace: kube-system +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: system:cloud-controller-manager +rules: +- apiGroups: + - "" + resources: + - nodes + verbs: + - '*' + +- apiGroups: + - "" + resources: + - nodes/status + verbs: + - patch + +- apiGroups: + - "" + resources: + - services + verbs: + - list + - watch + - patch + +- apiGroups: + - "" + resources: + - services/status + verbs: + - update + +- apiGroups: + - "" + resources: + - events + verbs: + - create + - patch + - update + +# For leader election +- apiGroups: + - "" + resources: + - endpoints + verbs: + - create + +- apiGroups: + - "" + resources: + - endpoints + resourceNames: + - "cloud-controller-manager" + verbs: + - get + - list + - watch + - update + +- apiGroups: + - "" + resources: + - configmaps + verbs: + - create + +- apiGroups: + - "" + resources: + - configmaps + resourceNames: + - "cloud-controller-manager" + verbs: + - get + - update + +- apiGroups: + - "" + resources: + - serviceaccounts + verbs: + - create +- apiGroups: + - "" + resources: + - secrets + verbs: + - get + - list + +# For the PVL +- apiGroups: + - "" + resources: + - persistentvolumes + verbs: + - list + - watch + - patch +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: oci-cloud-controller-manager +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: system:cloud-controller-manager +subjects: +- kind: ServiceAccount + name: cloud-controller-manager + namespace: kube-system diff --git a/kubespray/roles/kubernetes-apps/cluster_roles/tasks/main.yml b/kubespray/roles/kubernetes-apps/cluster_roles/tasks/main.yml new file mode 100644 index 0000000..fdb3205 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/cluster_roles/tasks/main.yml @@ -0,0 +1,87 @@ +--- +- name: Kubernetes Apps | Wait for kube-apiserver + uri: + url: "{{ kube_apiserver_endpoint }}/healthz" + validate_certs: no + client_cert: "{{ kube_apiserver_client_cert }}" + client_key: "{{ kube_apiserver_client_key }}" + register: result + until: result.status == 200 + retries: 10 + delay: 6 + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Apps | Add ClusterRoleBinding to admit nodes + template: + src: "node-crb.yml.j2" + dest: "{{ kube_config_dir }}/node-crb.yml" + mode: 0640 + register: node_crb_manifest + when: + - rbac_enabled + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Apply workaround to allow all nodes with cert O=system:nodes to register + kube: + name: "kubespray:system:node" + kubectl: "{{ bin_dir }}/kubectl" + resource: "clusterrolebinding" + filename: "{{ kube_config_dir }}/node-crb.yml" + state: latest + register: result + until: result is succeeded + retries: 10 + delay: 6 + when: + - rbac_enabled + - node_crb_manifest.changed + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Apps | Remove old webhook ClusterRole + kube: + name: "system:node-webhook" + kubectl: "{{ bin_dir }}/kubectl" + resource: "clusterrole" + state: absent + when: + - rbac_enabled + - inventory_hostname == groups['kube_control_plane'][0] + tags: node-webhook + +- name: Kubernetes Apps | Remove old webhook ClusterRoleBinding + kube: + name: "system:node-webhook" + kubectl: "{{ bin_dir }}/kubectl" + resource: "clusterrolebinding" + state: absent + when: + - rbac_enabled + - inventory_hostname == groups['kube_control_plane'][0] + tags: node-webhook + +- name: Configure Oracle Cloud provider + include_tasks: oci.yml + tags: oci + when: + - cloud_provider is defined + - cloud_provider == 'oci' + +- name: PriorityClass | Copy k8s-cluster-critical-pc.yml file + copy: + src: k8s-cluster-critical-pc.yml + dest: "{{ kube_config_dir }}/k8s-cluster-critical-pc.yml" + mode: 0640 + when: inventory_hostname == groups['kube_control_plane'] | last + +- name: PriorityClass | Create k8s-cluster-critical + kube: + name: k8s-cluster-critical + kubectl: "{{ bin_dir }}/kubectl" + resource: "PriorityClass" + filename: "{{ kube_config_dir }}/k8s-cluster-critical-pc.yml" + state: latest + register: result + until: result is succeeded + retries: 10 + delay: 6 + when: inventory_hostname == groups['kube_control_plane'] | last diff --git a/kubespray/roles/kubernetes-apps/cluster_roles/tasks/oci.yml b/kubespray/roles/kubernetes-apps/cluster_roles/tasks/oci.yml new file mode 100644 index 0000000..eb07463 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/cluster_roles/tasks/oci.yml @@ -0,0 +1,19 @@ +--- +- name: Copy OCI RBAC Manifest + copy: + src: "oci-rbac.yml" + dest: "{{ kube_config_dir }}/oci-rbac.yml" + mode: 0640 + when: + - cloud_provider is defined + - cloud_provider == 'oci' + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Apply OCI RBAC + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/oci-rbac.yml" + when: + - cloud_provider is defined + - cloud_provider == 'oci' + - inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/cluster_roles/templates/namespace.j2 b/kubespray/roles/kubernetes-apps/cluster_roles/templates/namespace.j2 new file mode 100644 index 0000000..f2e115a --- /dev/null +++ b/kubespray/roles/kubernetes-apps/cluster_roles/templates/namespace.j2 @@ -0,0 +1,4 @@ +apiVersion: v1 +kind: Namespace +metadata: + name: "kube-system" diff --git a/kubespray/roles/kubernetes-apps/cluster_roles/templates/node-crb.yml.j2 b/kubespray/roles/kubernetes-apps/cluster_roles/templates/node-crb.yml.j2 new file mode 100644 index 0000000..9a4a3c4 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/cluster_roles/templates/node-crb.yml.j2 @@ -0,0 +1,17 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + annotations: + rbac.authorization.kubernetes.io/autoupdate: "true" + labels: + kubernetes.io/bootstrapping: rbac-defaults + name: kubespray:system:node +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: system:node +subjects: +- apiGroup: rbac.authorization.k8s.io + kind: Group + name: system:nodes diff --git a/kubespray/roles/kubernetes-apps/cluster_roles/templates/vsphere-rbac.yml.j2 b/kubespray/roles/kubernetes-apps/cluster_roles/templates/vsphere-rbac.yml.j2 new file mode 100644 index 0000000..99da046 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/cluster_roles/templates/vsphere-rbac.yml.j2 @@ -0,0 +1,35 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: system:vsphere-cloud-provider +rules: +- apiGroups: + - "" + resources: + - nodes + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + - events + verbs: + - create + - patch + - update + +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: system:vsphere-cloud-provider +roleRef: + kind: ClusterRole + name: system:vsphere-cloud-provider + apiGroup: rbac.authorization.k8s.io +subjects: +- kind: ServiceAccount + name: vsphere-cloud-provider + namespace: kube-system diff --git a/kubespray/roles/kubernetes-apps/container_engine_accelerator/meta/main.yml b/kubespray/roles/kubernetes-apps/container_engine_accelerator/meta/main.yml new file mode 100644 index 0000000..c82c5d8 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_engine_accelerator/meta/main.yml @@ -0,0 +1,8 @@ +--- +dependencies: + - role: kubernetes-apps/container_engine_accelerator/nvidia_gpu + when: nvidia_accelerator_enabled + tags: + - apps + - nvidia_gpu + - container_engine_accelerator diff --git a/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/defaults/main.yml b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/defaults/main.yml new file mode 100644 index 0000000..6e870e4 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/defaults/main.yml @@ -0,0 +1,14 @@ +--- +nvidia_accelerator_enabled: false +nvidia_driver_version: "390.87" +nvidia_gpu_tesla_base_url: https://us.download.nvidia.com/tesla/ +nvidia_gpu_gtx_base_url: http://us.download.nvidia.com/XFree86/Linux-x86_64/ +nvidia_gpu_flavor: tesla +nvidia_url_end: "{{ nvidia_driver_version }}/NVIDIA-Linux-x86_64-{{ nvidia_driver_version }}.run" +nvidia_driver_install_container: false +nvidia_driver_install_centos_container: atzedevries/nvidia-centos-driver-installer:2 +nvidia_driver_install_ubuntu_container: registry.k8s.io/ubuntu-nvidia-driver-installer@sha256:7df76a0f0a17294e86f691c81de6bbb7c04a1b4b3d4ea4e7e2cccdc42e1f6d63 +nvidia_driver_install_supported: false +nvidia_gpu_device_plugin_container: "registry.k8s.io/nvidia-gpu-device-plugin@sha256:0842734032018be107fa2490c98156992911e3e1f2a21e059ff0105b07dd8e9e" +nvidia_gpu_nodes: [] +nvidia_gpu_device_plugin_memory: 30Mi diff --git a/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/tasks/main.yml b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/tasks/main.yml new file mode 100644 index 0000000..8cba9bf --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/tasks/main.yml @@ -0,0 +1,55 @@ +--- + +- name: Container Engine Acceleration Nvidia GPU | gather os specific variables + include_vars: "{{ item }}" + with_first_found: + - files: + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_release }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_major_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}.yml" + - "{{ ansible_os_family | lower }}.yml" + skip: true + +- name: Container Engine Acceleration Nvidia GPU | Set fact of download url Tesla + set_fact: + nvidia_driver_download_url_default: "{{ nvidia_gpu_tesla_base_url }}{{ nvidia_url_end }}" + when: nvidia_gpu_flavor | lower == "tesla" + +- name: Container Engine Acceleration Nvidia GPU | Set fact of download url GTX + set_fact: + nvidia_driver_download_url_default: "{{ nvidia_gpu_gtx_base_url }}{{ nvidia_url_end }}" + when: nvidia_gpu_flavor | lower == "gtx" + +- name: Container Engine Acceleration Nvidia GPU | Create addon dir + file: + path: "{{ kube_config_dir }}/addons/container_engine_accelerator" + owner: root + group: root + mode: 0755 + recurse: true + +- name: Container Engine Acceleration Nvidia GPU | Create manifests for nvidia accelerators + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/container_engine_accelerator/{{ item.file }}" + mode: 0644 + with_items: + - { name: nvidia-driver-install-daemonset, file: nvidia-driver-install-daemonset.yml, type: daemonset } + - { name: k8s-device-plugin-nvidia-daemonset, file: k8s-device-plugin-nvidia-daemonset.yml, type: daemonset } + register: container_engine_accelerator_manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] and nvidia_driver_install_container + +- name: Container Engine Acceleration Nvidia GPU | Apply manifests for nvidia accelerators + kube: + name: "{{ item.item.name }}" + namespace: "kube-system" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/container_engine_accelerator/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ container_engine_accelerator_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] and nvidia_driver_install_container and nvidia_driver_install_supported diff --git a/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/templates/k8s-device-plugin-nvidia-daemonset.yml.j2 b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/templates/k8s-device-plugin-nvidia-daemonset.yml.j2 new file mode 100644 index 0000000..c5a7f51 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/templates/k8s-device-plugin-nvidia-daemonset.yml.j2 @@ -0,0 +1,60 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: nvidia-gpu-device-plugin + namespace: kube-system + labels: + k8s-app: nvidia-gpu-device-plugin + addonmanager.kubernetes.io/mode: Reconcile +spec: + selector: + matchLabels: + k8s-app: nvidia-gpu-device-plugin + template: + metadata: + labels: + k8s-app: nvidia-gpu-device-plugin + spec: + priorityClassName: system-node-critical + affinity: + nodeAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: "nvidia.com/gpu" + operator: Exists + tolerations: + - operator: "Exists" + effect: "NoExecute" + - operator: "Exists" + effect: "NoSchedule" + hostNetwork: true + dnsPolicy: ClusterFirstWithHostNet + hostPID: true + volumes: + - name: device-plugin + hostPath: + path: /var/lib/kubelet/device-plugins + - name: dev + hostPath: + path: /dev + containers: + - image: "{{ nvidia_gpu_device_plugin_container }}" + command: ["/usr/bin/nvidia-gpu-device-plugin", "-logtostderr"] + name: nvidia-gpu-device-plugin + resources: + requests: + cpu: 50m + memory: {{ nvidia_gpu_device_plugin_memory }} + limits: + cpu: 50m + memory: {{ nvidia_gpu_device_plugin_memory }} + securityContext: + privileged: true + volumeMounts: + - name: device-plugin + mountPath: /device-plugin + - name: dev + mountPath: /dev + updateStrategy: + type: RollingUpdate diff --git a/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/templates/nvidia-driver-install-daemonset.yml.j2 b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/templates/nvidia-driver-install-daemonset.yml.j2 new file mode 100644 index 0000000..ea097ed --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/templates/nvidia-driver-install-daemonset.yml.j2 @@ -0,0 +1,82 @@ +# Copyright 2017 Google Inc. All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: nvidia-driver-installer + namespace: kube-system +spec: + selector: + matchLabels: + name: nvidia-driver-installer + template: + metadata: + labels: + name: nvidia-driver-installer + spec: + priorityClassName: system-node-critical + affinity: + nodeAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: "nvidia.com/gpu" + operator: Exists + tolerations: + - key: "nvidia.com/gpu" + effect: "NoSchedule" + operator: "Exists" + hostNetwork: true + dnsPolicy: ClusterFirstWithHostNet + hostPID: true + volumes: + - name: dev + hostPath: + path: /dev + - name: nvidia-install-dir-host + hostPath: + path: /home/kubernetes/bin/nvidia + - name: root-mount + hostPath: + path: / + initContainers: + - image: "{{ nvidia_driver_install_container }}" + name: nvidia-driver-installer + resources: + requests: + cpu: 0.15 + securityContext: + privileged: true + env: + - name: NVIDIA_INSTALL_DIR_HOST + value: /home/kubernetes/bin/nvidia + - name: NVIDIA_INSTALL_DIR_CONTAINER + value: /usr/local/nvidia + - name: ROOT_MOUNT_DIR + value: /root + - name: NVIDIA_DRIVER_VERSION + value: "{{ nvidia_driver_version }}" + - name: NVIDIA_DRIVER_DOWNLOAD_URL + value: "{{ nvidia_driver_download_url_default }}" + volumeMounts: + - name: nvidia-install-dir-host + mountPath: /usr/local/nvidia + - name: dev + mountPath: /dev + - name: root-mount + mountPath: /root + containers: + - image: "{{ pod_infra_image_repo }}:{{ pod_infra_image_tag }}" + name: pause diff --git a/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/vars/centos-7.yml b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/vars/centos-7.yml new file mode 100644 index 0000000..b1ea65b --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/vars/centos-7.yml @@ -0,0 +1,3 @@ +--- +nvidia_driver_install_container: "{{ nvidia_driver_install_centos_container }}" +nvidia_driver_install_supported: true diff --git a/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/vars/ubuntu-16.yml b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/vars/ubuntu-16.yml new file mode 100644 index 0000000..f1bfdfc --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/vars/ubuntu-16.yml @@ -0,0 +1,3 @@ +--- +nvidia_driver_install_container: "{{ nvidia_driver_install_ubuntu_container }}" +nvidia_driver_install_supported: true diff --git a/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/vars/ubuntu-18.yml b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/vars/ubuntu-18.yml new file mode 100644 index 0000000..f1bfdfc --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_engine_accelerator/nvidia_gpu/vars/ubuntu-18.yml @@ -0,0 +1,3 @@ +--- +nvidia_driver_install_container: "{{ nvidia_driver_install_ubuntu_container }}" +nvidia_driver_install_supported: true diff --git a/kubespray/roles/kubernetes-apps/container_runtimes/crun/tasks/main.yaml b/kubespray/roles/kubernetes-apps/container_runtimes/crun/tasks/main.yaml new file mode 100644 index 0000000..6690141 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_runtimes/crun/tasks/main.yaml @@ -0,0 +1,19 @@ +--- + +- name: Crun | Copy runtime class manifest + template: + src: runtimeclass-crun.yml + dest: "{{ kube_config_dir }}/runtimeclass-crun.yml" + mode: "0664" + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Crun | Apply manifests + kube: + name: "runtimeclass-crun" + kubectl: "{{ bin_dir }}/kubectl" + resource: "runtimeclass" + filename: "{{ kube_config_dir }}/runtimeclass-crun.yml" + state: "latest" + when: + - inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/container_runtimes/crun/templates/runtimeclass-crun.yml b/kubespray/roles/kubernetes-apps/container_runtimes/crun/templates/runtimeclass-crun.yml new file mode 100644 index 0000000..99d97e6 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_runtimes/crun/templates/runtimeclass-crun.yml @@ -0,0 +1,6 @@ +--- +kind: RuntimeClass +apiVersion: node.k8s.io/v1 +metadata: + name: crun +handler: crun diff --git a/kubespray/roles/kubernetes-apps/container_runtimes/gvisor/tasks/main.yaml b/kubespray/roles/kubernetes-apps/container_runtimes/gvisor/tasks/main.yaml new file mode 100644 index 0000000..90562f2 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_runtimes/gvisor/tasks/main.yaml @@ -0,0 +1,34 @@ +--- +- name: GVisor | Create addon dir + file: + path: "{{ kube_config_dir }}/addons/gvisor" + owner: root + group: root + mode: 0755 + recurse: true + +- name: GVisor | Templates List + set_fact: + gvisor_templates: + - { name: runtimeclass-gvisor, file: runtimeclass-gvisor.yml, type: runtimeclass } + +- name: GVisort | Create manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/gvisor/{{ item.file }}" + mode: 0644 + with_items: "{{ gvisor_templates }}" + register: gvisor_manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: GVisor | Apply manifests + kube: + name: "{{ item.item.name }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/gvisor/{{ item.item.file }}" + state: "latest" + with_items: "{{ gvisor_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/container_runtimes/gvisor/templates/runtimeclass-gvisor.yml.j2 b/kubespray/roles/kubernetes-apps/container_runtimes/gvisor/templates/runtimeclass-gvisor.yml.j2 new file mode 100644 index 0000000..64465fa --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_runtimes/gvisor/templates/runtimeclass-gvisor.yml.j2 @@ -0,0 +1,6 @@ +--- +kind: RuntimeClass +apiVersion: node.k8s.io/v1 +metadata: + name: gvisor +handler: runsc diff --git a/kubespray/roles/kubernetes-apps/container_runtimes/kata_containers/defaults/main.yaml b/kubespray/roles/kubernetes-apps/container_runtimes/kata_containers/defaults/main.yaml new file mode 100644 index 0000000..6eacb79 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_runtimes/kata_containers/defaults/main.yaml @@ -0,0 +1,5 @@ +--- + +kata_containers_qemu_overhead: true +kata_containers_qemu_overhead_fixed_cpu: 250m +kata_containers_qemu_overhead_fixed_memory: 160Mi diff --git a/kubespray/roles/kubernetes-apps/container_runtimes/kata_containers/tasks/main.yaml b/kubespray/roles/kubernetes-apps/container_runtimes/kata_containers/tasks/main.yaml new file mode 100644 index 0000000..a07c7c2 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_runtimes/kata_containers/tasks/main.yaml @@ -0,0 +1,35 @@ +--- + +- name: Kata Containers | Create addon dir + file: + path: "{{ kube_config_dir }}/addons/kata_containers" + owner: root + group: root + mode: 0755 + recurse: true + +- name: Kata Containers | Templates list + set_fact: + kata_containers_templates: + - { name: runtimeclass-kata-qemu, file: runtimeclass-kata-qemu.yml, type: runtimeclass } + +- name: Kata Containers | Create manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/kata_containers/{{ item.file }}" + mode: 0644 + with_items: "{{ kata_containers_templates }}" + register: kata_containers_manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kata Containers | Apply manifests + kube: + name: "{{ item.item.name }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/kata_containers/{{ item.item.file }}" + state: "latest" + with_items: "{{ kata_containers_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/container_runtimes/kata_containers/templates/runtimeclass-kata-qemu.yml.j2 b/kubespray/roles/kubernetes-apps/container_runtimes/kata_containers/templates/runtimeclass-kata-qemu.yml.j2 new file mode 100644 index 0000000..2240cdb --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_runtimes/kata_containers/templates/runtimeclass-kata-qemu.yml.j2 @@ -0,0 +1,12 @@ +--- +kind: RuntimeClass +apiVersion: node.k8s.io/v1 +metadata: + name: kata-qemu +handler: kata-qemu +{% if kata_containers_qemu_overhead %} +overhead: + podFixed: + cpu: {{ kata_containers_qemu_overhead_fixed_cpu }} + memory: {{ kata_containers_qemu_overhead_fixed_memory }} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/container_runtimes/meta/main.yml b/kubespray/roles/kubernetes-apps/container_runtimes/meta/main.yml new file mode 100644 index 0000000..8584117 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_runtimes/meta/main.yml @@ -0,0 +1,31 @@ +--- +dependencies: + - role: kubernetes-apps/container_runtimes/kata_containers + when: kata_containers_enabled + tags: + - apps + - kata-containers + - container-runtimes + + - role: kubernetes-apps/container_runtimes/gvisor + when: gvisor_enabled + tags: + - apps + - gvisor + - container-runtimes + + - role: kubernetes-apps/container_runtimes/crun + when: crun_enabled + tags: + - apps + - crun + - container-runtimes + + - role: kubernetes-apps/container_runtimes/youki + when: + - youki_enabled + - container_manager == 'crio' + tags: + - apps + - youki + - container-runtimes diff --git a/kubespray/roles/kubernetes-apps/container_runtimes/youki/tasks/main.yaml b/kubespray/roles/kubernetes-apps/container_runtimes/youki/tasks/main.yaml new file mode 100644 index 0000000..8ba7c7a --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_runtimes/youki/tasks/main.yaml @@ -0,0 +1,19 @@ +--- + +- name: Youki | Copy runtime class manifest + template: + src: runtimeclass-youki.yml + dest: "{{ kube_config_dir }}/runtimeclass-youki.yml" + mode: "0664" + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Youki | Apply manifests + kube: + name: "runtimeclass-youki" + kubectl: "{{ bin_dir }}/kubectl" + resource: "runtimeclass" + filename: "{{ kube_config_dir }}/runtimeclass-youki.yml" + state: "latest" + when: + - inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/container_runtimes/youki/templates/runtimeclass-youki.yml b/kubespray/roles/kubernetes-apps/container_runtimes/youki/templates/runtimeclass-youki.yml new file mode 100644 index 0000000..b68bd06 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/container_runtimes/youki/templates/runtimeclass-youki.yml @@ -0,0 +1,6 @@ +--- +kind: RuntimeClass +apiVersion: node.k8s.io/v1 +metadata: + name: youki +handler: youki diff --git a/kubespray/roles/kubernetes-apps/csi_driver/OWNERS b/kubespray/roles/kubernetes-apps/csi_driver/OWNERS new file mode 100644 index 0000000..6cfbaa8 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/OWNERS @@ -0,0 +1,6 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +approvers: +reviewers: + - alijahnas + - luckySB diff --git a/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/defaults/main.yml b/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/defaults/main.yml new file mode 100644 index 0000000..33df37c --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/defaults/main.yml @@ -0,0 +1,11 @@ +--- +aws_ebs_csi_enable_volume_scheduling: true +aws_ebs_csi_enable_volume_snapshot: false +aws_ebs_csi_enable_volume_resizing: false +aws_ebs_csi_controller_replicas: 1 +aws_ebs_csi_plugin_image_tag: latest + +# Add annotions to ebs_csi_controller. Useful if using kube2iam for role assumption +# aws_ebs_csi_annotations: +# - key: iam.amazonaws.com/role +# value: your-ebs-role-arn diff --git a/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/tasks/main.yml b/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/tasks/main.yml new file mode 100644 index 0000000..5570dcc --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/tasks/main.yml @@ -0,0 +1,26 @@ +--- +- name: AWS CSI Driver | Generate Manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: aws-ebs-csi-driver, file: aws-ebs-csi-driver.yml} + - {name: aws-ebs-csi-controllerservice, file: aws-ebs-csi-controllerservice-rbac.yml} + - {name: aws-ebs-csi-controllerservice, file: aws-ebs-csi-controllerservice.yml} + - {name: aws-ebs-csi-nodeservice, file: aws-ebs-csi-nodeservice.yml} + register: aws_csi_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: AWS CSI Driver | Apply Manifests + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ aws_csi_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item.file }}" diff --git a/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-controllerservice-rbac.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-controllerservice-rbac.yml.j2 new file mode 100644 index 0000000..87bfa31 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-controllerservice-rbac.yml.j2 @@ -0,0 +1,180 @@ +# Controller Service +apiVersion: v1 +kind: ServiceAccount +metadata: + name: ebs-csi-controller-sa + namespace: kube-system + +--- + +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: ebs-external-provisioner-role +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["list", "watch", "create", "delete"] + - apiGroups: [""] + resources: ["persistentvolumeclaims"] + verbs: ["get", "list", "watch", "update"] + - apiGroups: ["storage.k8s.io"] + resources: ["storageclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["get", "list", "watch", "create", "update", "patch"] + - apiGroups: ["storage.k8s.io"] + resources: ["csinodes"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["nodes"] + verbs: ["get", "list", "watch"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "watch", "list", "delete", "update", "create"] + +--- + +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: ebs-csi-provisioner-binding +subjects: + - kind: ServiceAccount + name: ebs-csi-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: ebs-external-provisioner-role + apiGroup: rbac.authorization.k8s.io + +--- + +# The permissions in this ClusterRole are tightly coupled with the version of csi-attacher used. More information about this can be found in kubernetes-csi/external-attacher. +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: ebs-external-attacher-role +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "patch"] + - apiGroups: ["storage.k8s.io"] + resources: ["csinodes"] + verbs: ["get", "list", "watch"] + - apiGroups: ["storage.k8s.io"] + resources: ["volumeattachments"] + verbs: ["get", "list", "watch", "patch"] + - apiGroups: ["storage.k8s.io"] + resources: ["volumeattachments/status"] + verbs: ["patch"] + +--- + +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: ebs-csi-attacher-binding +subjects: + - kind: ServiceAccount + name: ebs-csi-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: ebs-external-attacher-role + apiGroup: rbac.authorization.k8s.io + +{% if aws_ebs_csi_enable_volume_snapshot %} +--- + +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: ebs-external-snapshotter-role +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["persistentvolumeclaims"] + verbs: ["get", "list", "watch"] + - apiGroups: ["storage.k8s.io"] + resources: ["storageclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["list", "watch", "create", "update", "patch"] + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotcontents"] + verbs: ["create", "get", "list", "watch", "update", "delete"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshots"] + verbs: ["get", "list", "watch", "update"] + - apiGroups: ["apiextensions.k8s.io"] + resources: ["customresourcedefinitions"] + verbs: ["create", "list", "watch", "delete"] + +--- + +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: ebs-csi-snapshotter-binding +subjects: + - kind: ServiceAccount + name: ebs-csi-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: ebs-external-snapshotter-role + apiGroup: rbac.authorization.k8s.io + +{% endif %} + +{% if aws_ebs_csi_enable_volume_resizing %} +--- + +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: ebs-external-resizer-role +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "update", "patch"] + - apiGroups: [""] + resources: ["persistentvolumeclaims"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["persistentvolumeclaims/status"] + verbs: ["update", "patch"] + - apiGroups: ["storage.k8s.io"] + resources: ["storageclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["list", "watch", "create", "update", "patch"] + +--- + +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: ebs-csi-resizer-binding +subjects: + - kind: ServiceAccount + name: ebs-csi-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: ebs-external-resizer-role + apiGroup: rbac.authorization.k8s.io + +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-controllerservice.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-controllerservice.yml.j2 new file mode 100644 index 0000000..d58490a --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-controllerservice.yml.j2 @@ -0,0 +1,131 @@ +--- +kind: Deployment +apiVersion: apps/v1 +metadata: + name: ebs-csi-controller + namespace: kube-system +spec: + replicas: {{ aws_ebs_csi_controller_replicas }} + selector: + matchLabels: + app: ebs-csi-controller + app.kubernetes.io/name: aws-ebs-csi-driver + template: + metadata: + labels: + app: ebs-csi-controller + app.kubernetes.io/name: aws-ebs-csi-driver +{% if aws_ebs_csi_annotations is defined %} + annotations: +{% for annotation in aws_ebs_csi_annotations %} + {{ annotation.key }}: {{ annotation.value }} +{% endfor %} +{% endif %} + spec: + nodeSelector: + kubernetes.io/os: linux + serviceAccountName: ebs-csi-controller-sa + priorityClassName: system-cluster-critical + containers: + - name: ebs-plugin + image: {{ aws_ebs_csi_plugin_image_repo }}:{{ aws_ebs_csi_plugin_image_tag }} + args: + - --endpoint=$(CSI_ENDPOINT) +{% if aws_ebs_csi_extra_volume_tags is defined %} + - --extra-volume-tags={{ aws_ebs_csi_extra_volume_tags }} +{% endif %} + - --logtostderr + - --v=5 + env: + - name: CSI_ENDPOINT + value: unix:///var/lib/csi/sockets/pluginproxy/csi.sock + - name: AWS_ACCESS_KEY_ID + valueFrom: + secretKeyRef: + name: aws-secret + key: key_id + optional: true + - name: AWS_SECRET_ACCESS_KEY + valueFrom: + secretKeyRef: + name: aws-secret + key: access_key + optional: true + volumeMounts: + - name: socket-dir + mountPath: /var/lib/csi/sockets/pluginproxy/ + ports: + - name: healthz + containerPort: 9808 + protocol: TCP + livenessProbe: + httpGet: + path: /healthz + port: healthz + initialDelaySeconds: 10 + timeoutSeconds: 3 + periodSeconds: 10 + failureThreshold: 5 + - name: csi-provisioner + image: {{ csi_provisioner_image_repo }}:{{ csi_provisioner_image_tag }} + args: + - --csi-address=$(ADDRESS) + - --v=5 +{% if aws_ebs_csi_enable_volume_scheduling %} + - --feature-gates=Topology=true +{% endif %} + - --leader-election=true + env: + - name: ADDRESS + value: /var/lib/csi/sockets/pluginproxy/csi.sock + volumeMounts: + - name: socket-dir + mountPath: /var/lib/csi/sockets/pluginproxy/ + - name: csi-attacher + image: {{ csi_attacher_image_repo }}:{{ csi_attacher_image_tag }} + args: + - --csi-address=$(ADDRESS) + - --v=5 + env: + - name: ADDRESS + value: /var/lib/csi/sockets/pluginproxy/csi.sock + volumeMounts: + - name: socket-dir + mountPath: /var/lib/csi/sockets/pluginproxy/ +{% if aws_ebs_csi_enable_volume_snapshot %} + - name: csi-snapshotter + image: {{ csi_snapshotter_image_repo }}:{{ csi_snapshotter_image_tag }} + args: + - --csi-address=$(ADDRESS) + - --timeout=15s + env: + - name: ADDRESS + value: /var/lib/csi/sockets/pluginproxy/csi.sock + volumeMounts: + - name: socket-dir + mountPath: /var/lib/csi/sockets/pluginproxy/ +{% endif %} +{% if aws_ebs_csi_enable_volume_resizing %} + - name: csi-resizer + image: {{ csi_resizer_image_repo }}:{{ csi_resizer_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - --csi-address=$(ADDRESS) + - --v=5 + env: + - name: ADDRESS + value: /var/lib/csi/sockets/pluginproxy/csi.sock + volumeMounts: + - name: socket-dir + mountPath: /var/lib/csi/sockets/pluginproxy/ +{% endif %} + - name: liveness-probe + image: {{ csi_livenessprobe_image_repo }}:{{ csi_livenessprobe_image_tag }} + args: + - --csi-address=/csi/csi.sock + volumeMounts: + - name: socket-dir + mountPath: /csi + volumes: + - name: socket-dir + emptyDir: {} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-driver.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-driver.yml.j2 new file mode 100644 index 0000000..99c6c5b --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-driver.yml.j2 @@ -0,0 +1,8 @@ +--- +apiVersion: storage.k8s.io/v1 +kind: CSIDriver +metadata: + name: ebs.csi.aws.com +spec: + attachRequired: true + podInfoOnMount: false diff --git a/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-nodeservice.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-nodeservice.yml.j2 new file mode 100644 index 0000000..1dc1925 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/aws_ebs/templates/aws-ebs-csi-nodeservice.yml.j2 @@ -0,0 +1,101 @@ +--- +# Node Service +kind: DaemonSet +apiVersion: apps/v1 +metadata: + name: ebs-csi-node + namespace: kube-system +spec: + selector: + matchLabels: + app: ebs-csi-node + app.kubernetes.io/name: aws-ebs-csi-driver + template: + metadata: + labels: + app: ebs-csi-node + app.kubernetes.io/name: aws-ebs-csi-driver + spec: + nodeSelector: + kubernetes.io/os: linux + hostNetwork: true + priorityClassName: system-node-critical + containers: + - name: ebs-plugin + securityContext: + privileged: true + image: {{ aws_ebs_csi_plugin_image_repo }}:{{ aws_ebs_csi_plugin_image_tag }} + args: + - --endpoint=$(CSI_ENDPOINT) +{% if aws_ebs_csi_extra_volume_tags is defined %} + - --extra-volume-tags={{ aws_ebs_csi_extra_volume_tags }} +{% endif %} + - --logtostderr + - --v=5 + env: + - name: CSI_ENDPOINT + value: unix:/csi/csi.sock + volumeMounts: + - name: kubelet-dir + mountPath: /var/lib/kubelet + mountPropagation: "Bidirectional" + - name: plugin-dir + mountPath: /csi + - name: device-dir + mountPath: /dev + ports: + - name: healthz + containerPort: 9808 + protocol: TCP + livenessProbe: + httpGet: + path: /healthz + port: healthz + initialDelaySeconds: 10 + timeoutSeconds: 3 + periodSeconds: 10 + failureThreshold: 5 + - name: node-driver-registrar + image: {{ csi_node_driver_registrar_image_repo }}:{{ csi_node_driver_registrar_image_tag }} + args: + - --csi-address=$(ADDRESS) + - --kubelet-registration-path=$(DRIVER_REG_SOCK_PATH) + - --v=5 + lifecycle: + preStop: + exec: + command: ["/bin/sh", "-c", "rm -rf /registration/ebs.csi.aws.com-reg.sock /csi/csi.sock"] + env: + - name: ADDRESS + value: /csi/csi.sock + - name: DRIVER_REG_SOCK_PATH + value: /var/lib/kubelet/plugins/ebs.csi.aws.com/csi.sock + volumeMounts: + - name: plugin-dir + mountPath: /csi + - name: registration-dir + mountPath: /registration + - name: liveness-probe + image: {{ csi_livenessprobe_image_repo }}:{{ csi_livenessprobe_image_tag }} + args: + - --csi-address=/csi/csi.sock + volumeMounts: + - name: plugin-dir + mountPath: /csi + volumes: + - name: kubelet-dir + hostPath: + path: /var/lib/kubelet + type: Directory + - name: plugin-dir + hostPath: + path: /var/lib/kubelet/plugins/ebs.csi.aws.com/ + type: DirectoryOrCreate + - name: registration-dir + hostPath: + path: /var/lib/kubelet/plugins_registry/ + type: Directory + - name: device-dir + hostPath: + path: /dev + type: Directory diff --git a/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/defaults/main.yml b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/defaults/main.yml new file mode 100644 index 0000000..341cc97 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/defaults/main.yml @@ -0,0 +1,6 @@ +--- +azure_csi_use_instance_metadata: true +azure_csi_controller_replicas: 2 +azure_csi_plugin_image_tag: latest +azure_csi_controller_affinity: {} +azure_csi_node_affinity: {} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/tasks/azure-credential-check.yml b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/tasks/azure-credential-check.yml new file mode 100644 index 0000000..0a858ee --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/tasks/azure-credential-check.yml @@ -0,0 +1,54 @@ +--- +- name: Azure CSI Driver | check azure_csi_tenant_id value + fail: + msg: "azure_csi_tenant_id is missing" + when: azure_csi_tenant_id is not defined or not azure_csi_tenant_id + +- name: Azure CSI Driver | check azure_csi_subscription_id value + fail: + msg: "azure_csi_subscription_id is missing" + when: azure_csi_subscription_id is not defined or not azure_csi_subscription_id + +- name: Azure CSI Driver | check azure_csi_aad_client_id value + fail: + msg: "azure_csi_aad_client_id is missing" + when: azure_csi_aad_client_id is not defined or not azure_csi_aad_client_id + +- name: Azure CSI Driver | check azure_csi_aad_client_secret value + fail: + msg: "azure_csi_aad_client_secret is missing" + when: azure_csi_aad_client_secret is not defined or not azure_csi_aad_client_secret + +- name: Azure CSI Driver | check azure_csi_resource_group value + fail: + msg: "azure_csi_resource_group is missing" + when: azure_csi_resource_group is not defined or not azure_csi_resource_group + +- name: Azure CSI Driver | check azure_csi_location value + fail: + msg: "azure_csi_location is missing" + when: azure_csi_location is not defined or not azure_csi_location + +- name: Azure CSI Driver | check azure_csi_subnet_name value + fail: + msg: "azure_csi_subnet_name is missing" + when: azure_csi_subnet_name is not defined or not azure_csi_subnet_name + +- name: Azure CSI Driver | check azure_csi_security_group_name value + fail: + msg: "azure_csi_security_group_name is missing" + when: azure_csi_security_group_name is not defined or not azure_csi_security_group_name + +- name: Azure CSI Driver | check azure_csi_vnet_name value + fail: + msg: "azure_csi_vnet_name is missing" + when: azure_csi_vnet_name is not defined or not azure_csi_vnet_name + +- name: Azure CSI Driver | check azure_csi_vnet_resource_group value + fail: + msg: "azure_csi_vnet_resource_group is missing" + when: azure_csi_vnet_resource_group is not defined or not azure_csi_vnet_resource_group + +- name: "Azure CSI Driver | check azure_csi_use_instance_metadata is a bool" + assert: + that: azure_csi_use_instance_metadata | type_debug == 'bool' diff --git a/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/tasks/main.yml b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/tasks/main.yml new file mode 100644 index 0000000..a94656f --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/tasks/main.yml @@ -0,0 +1,45 @@ +--- +- name: Azure CSI Driver | Check Azure credentials + include_tasks: azure-credential-check.yml + +- name: Azure CSI Driver | Write Azure CSI cloud-config + template: + src: "azure-csi-cloud-config.j2" + dest: "{{ kube_config_dir }}/azure_csi_cloud_config" + group: "{{ kube_cert_group }}" + mode: 0640 + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Azure CSI Driver | Get base64 cloud-config + slurp: + src: "{{ kube_config_dir }}/azure_csi_cloud_config" + register: cloud_config_secret + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Azure CSI Driver | Generate Manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: azure-csi-azuredisk-driver, file: azure-csi-azuredisk-driver.yml} + - {name: azure-csi-cloud-config-secret, file: azure-csi-cloud-config-secret.yml} + - {name: azure-csi-azuredisk-controller, file: azure-csi-azuredisk-controller-rbac.yml} + - {name: azure-csi-azuredisk-controller, file: azure-csi-azuredisk-controller.yml} + - {name: azure-csi-azuredisk-node-rbac, file: azure-csi-azuredisk-node-rbac.yml} + - {name: azure-csi-azuredisk-node, file: azure-csi-azuredisk-node.yml} + register: azure_csi_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Azure CSI Driver | Apply Manifests + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ azure_csi_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item.file }}" diff --git a/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-controller-rbac.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-controller-rbac.yml.j2 new file mode 100644 index 0000000..16f4c98 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-controller-rbac.yml.j2 @@ -0,0 +1,230 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: csi-azuredisk-controller-sa + namespace: kube-system +--- + +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: azuredisk-external-provisioner-role +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "create", "delete"] + - apiGroups: [""] + resources: ["persistentvolumeclaims"] + verbs: ["get", "list", "watch", "update"] + - apiGroups: ["storage.k8s.io"] + resources: ["storageclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["get", "list", "watch", "create", "update", "patch"] + - apiGroups: ["storage.k8s.io"] + resources: ["csinodes"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["nodes"] + verbs: ["get", "list", "watch"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "list", "watch", "create", "update", "patch"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshots"] + verbs: ["get", "list"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotcontents"] + verbs: ["get", "list"] +--- + +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: azuredisk-csi-provisioner-binding +subjects: + - kind: ServiceAccount + name: csi-azuredisk-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: azuredisk-external-provisioner-role + apiGroup: rbac.authorization.k8s.io + +--- + +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: azuredisk-external-attacher-role +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "update"] + - apiGroups: [""] + resources: ["nodes"] + verbs: ["get", "list", "watch"] + - apiGroups: ["csi.storage.k8s.io"] + resources: ["csinodeinfos"] + verbs: ["get", "list", "watch"] + - apiGroups: ["storage.k8s.io"] + resources: ["volumeattachments"] + verbs: ["get", "list", "watch", "update", "patch"] + - apiGroups: ["storage.k8s.io"] + resources: ["volumeattachments/status"] + verbs: ["get", "list", "watch", "update", "patch"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "list", "watch", "create", "update", "patch"] +--- + +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: azuredisk-csi-attacher-binding +subjects: + - kind: ServiceAccount + name: csi-azuredisk-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: azuredisk-external-attacher-role + apiGroup: rbac.authorization.k8s.io + +--- + +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: azuredisk-cluster-driver-registrar-role +rules: + - apiGroups: ["apiextensions.k8s.io"] + resources: ["customresourcedefinitions"] + verbs: ["create", "list", "watch", "delete"] + - apiGroups: ["csi.storage.k8s.io"] + resources: ["csidrivers"] + verbs: ["create", "delete"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "list", "watch", "create", "update", "patch"] +--- + +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: azuredisk-csi-driver-registrar-binding +subjects: + - kind: ServiceAccount + name: csi-azuredisk-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: azuredisk-cluster-driver-registrar-role + apiGroup: rbac.authorization.k8s.io + +--- + +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: azuredisk-external-snapshotter-role +rules: + - apiGroups: [""] + resources: ["events"] + verbs: ["list", "watch", "create", "update", "patch"] + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotcontents"] + verbs: ["create", "get", "list", "watch", "update", "delete"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotcontents/status"] + verbs: ["update"] + - apiGroups: ["apiextensions.k8s.io"] + resources: ["customresourcedefinitions"] + verbs: ["create", "list", "watch", "delete"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "watch", "list", "delete", "update", "create"] +--- + +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: azuredisk-csi-snapshotter-binding +subjects: + - kind: ServiceAccount + name: csi-azuredisk-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: azuredisk-external-snapshotter-role + apiGroup: rbac.authorization.k8s.io +--- + +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: azuredisk-external-resizer-role +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "update", "patch"] + - apiGroups: [""] + resources: ["persistentvolumeclaims"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["persistentvolumeclaims/status"] + verbs: ["update", "patch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["list", "watch", "create", "update", "patch"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "list", "watch", "create", "update", "patch"] + - apiGroups: [""] + resources: ["pods"] + verbs: ["get", "list", "watch"] +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: azuredisk-csi-resizer-role +subjects: + - kind: ServiceAccount + name: csi-azuredisk-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: azuredisk-external-resizer-role + apiGroup: rbac.authorization.k8s.io + +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-azuredisk-controller-secret-role +rules: + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list"] + +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-azuredisk-controller-secret-binding +subjects: + - kind: ServiceAccount + name: csi-azuredisk-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-azuredisk-controller-secret-role + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-controller.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-controller.yml.j2 new file mode 100644 index 0000000..36d38ac --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-controller.yml.j2 @@ -0,0 +1,179 @@ +--- +kind: Deployment +apiVersion: apps/v1 +metadata: + name: csi-azuredisk-controller + namespace: kube-system +spec: + replicas: {{ azure_csi_controller_replicas }} + selector: + matchLabels: + app: csi-azuredisk-controller + template: + metadata: + labels: + app: csi-azuredisk-controller + spec: + hostNetwork: true + serviceAccountName: csi-azuredisk-controller-sa + nodeSelector: + kubernetes.io/os: linux + priorityClassName: system-cluster-critical + tolerations: + - key: "node-role.kubernetes.io/master" + effect: "NoSchedule" + - key: "node-role.kubernetes.io/control-plane" + effect: "NoSchedule" +{% if azure_csi_controller_affinity %} + affinity: + {{ azure_csi_controller_affinity | to_nice_yaml | indent(width=8) }} +{% endif %} + containers: + - name: csi-provisioner + image: {{ azure_csi_image_repo }}/csi-provisioner:{{ azure_csi_provisioner_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - "--feature-gates=Topology=true" + - "--csi-address=$(ADDRESS)" + - "--v=2" + - "--timeout=15s" + - "--leader-election" + - "--worker-threads=40" + - "--extra-create-metadata=true" + - "--strict-topology=true" + env: + - name: ADDRESS + value: /csi/csi.sock + volumeMounts: + - mountPath: /csi + name: socket-dir + resources: + limits: + memory: 500Mi + requests: + cpu: 10m + memory: 20Mi + - name: csi-attacher + image: {{ azure_csi_image_repo }}/csi-attacher:{{ azure_csi_attacher_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - "-v=2" + - "-csi-address=$(ADDRESS)" + - "-timeout=600s" + - "-leader-election" + - "-worker-threads=500" + env: + - name: ADDRESS + value: /csi/csi.sock + volumeMounts: + - mountPath: /csi + name: socket-dir + resources: + limits: + memory: 500Mi + requests: + cpu: 10m + memory: 20Mi + - name: csi-snapshotter + image: {{ azure_csi_image_repo }}/csi-snapshotter:{{ azure_csi_snapshotter_image_tag }} + args: + - "-csi-address=$(ADDRESS)" + - "-leader-election" + - "-v=2" + env: + - name: ADDRESS + value: /csi/csi.sock + volumeMounts: + - name: socket-dir + mountPath: /csi + resources: + limits: + memory: 100Mi + requests: + cpu: 10m + memory: 20Mi + - name: csi-resizer + image: {{ azure_csi_image_repo }}/csi-resizer:{{ azure_csi_resizer_image_tag }} + args: + - "-csi-address=$(ADDRESS)" + - "-v=2" + - "-leader-election" + - '-handle-volume-inuse-error=false' + - "-timeout=60s" + env: + - name: ADDRESS + value: /csi/csi.sock + volumeMounts: + - name: socket-dir + mountPath: /csi + resources: + limits: + memory: 500Mi + requests: + cpu: 10m + memory: 20Mi + - name: liveness-probe + image: {{ azure_csi_image_repo }}/livenessprobe:{{ azure_csi_livenessprobe_image_tag }} + args: + - --csi-address=/csi/csi.sock + - --probe-timeout=3s + - --health-port=29602 + - --v=2 + volumeMounts: + - name: socket-dir + mountPath: /csi + resources: + limits: + memory: 100Mi + requests: + cpu: 10m + memory: 20Mi + - name: azuredisk + image: {{ azure_csi_plugin_image_repo }}/azuredisk-csi:{{ azure_csi_plugin_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - "--v=5" + - "--endpoint=$(CSI_ENDPOINT)" + - "--metrics-address=0.0.0.0:29604" + - "--disable-avset-nodes=true" + - "--drivername=disk.csi.azure.com" + - "--cloud-config-secret-name=cloud-config" + - "--cloud-config-secret-namespace=kube-system" + ports: + - containerPort: 29602 + name: healthz + protocol: TCP + - containerPort: 29604 + name: metrics + protocol: TCP + livenessProbe: + failureThreshold: 5 + httpGet: + path: /healthz + port: healthz + initialDelaySeconds: 30 + timeoutSeconds: 10 + periodSeconds: 30 + env: + - name: AZURE_CREDENTIAL_FILE + value: "/etc/kubernetes/azure.json" + - name: CSI_ENDPOINT + value: unix:///csi/csi.sock + volumeMounts: + - mountPath: /csi + name: socket-dir + - mountPath: /etc/kubernetes/ + name: azure-cred + readOnly: true + resources: + limits: + memory: 500Mi + requests: + cpu: 10m + memory: 20Mi + volumes: + - name: socket-dir + emptyDir: {} + - name: azure-cred + secret: + secretName: cloud-config diff --git a/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-driver.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-driver.yml.j2 new file mode 100644 index 0000000..c7cba34 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-driver.yml.j2 @@ -0,0 +1,10 @@ +--- +apiVersion: storage.k8s.io/v1 +kind: CSIDriver +metadata: + name: disk.csi.azure.com +spec: + attachRequired: true + podInfoOnMount: true + volumeLifecycleModes: # added in Kubernetes 1.16 + - Persistent diff --git a/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-node-rbac.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-node-rbac.yml.j2 new file mode 100644 index 0000000..d55ea0d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-node-rbac.yml.j2 @@ -0,0 +1,30 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: csi-azuredisk-node-sa + namespace: kube-system + +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-azuredisk-node-secret-role +rules: + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list"] + +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-azuredisk-node-secret-binding +subjects: + - kind: ServiceAccount + name: csi-azuredisk-node-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-azuredisk-node-secret-role + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-node.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-node.yml.j2 new file mode 100644 index 0000000..4d80319 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-azuredisk-node.yml.j2 @@ -0,0 +1,168 @@ +--- +kind: DaemonSet +apiVersion: apps/v1 +metadata: + name: csi-azuredisk-node + namespace: kube-system +spec: + updateStrategy: + rollingUpdate: + maxUnavailable: 1 + type: RollingUpdate + selector: + matchLabels: + app: csi-azuredisk-node + template: + metadata: + labels: + app: csi-azuredisk-node + spec: + hostNetwork: true + dnsPolicy: Default + serviceAccountName: csi-azuredisk-node-sa + nodeSelector: + kubernetes.io/os: linux +{% if azure_csi_node_affinity %} + affinity: + {{ azure_csi_node_affinity | to_nice_yaml | indent(width=8) }} +{% endif %} + priorityClassName: system-node-critical + tolerations: + - operator: Exists + containers: + - name: liveness-probe + volumeMounts: + - mountPath: /csi + name: socket-dir + image: {{ azure_csi_image_repo }}/livenessprobe:{{ azure_csi_livenessprobe_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - --csi-address=/csi/csi.sock + - --probe-timeout=3s + - --health-port=29603 + - --v=2 + resources: + limits: + memory: 100Mi + requests: + cpu: 10m + memory: 20Mi + - name: node-driver-registrar + image: {{ azure_csi_image_repo }}/csi-node-driver-registrar:{{ azure_csi_node_registrar_image_tag }} + args: + - --csi-address=$(ADDRESS) + - --kubelet-registration-path=$(DRIVER_REG_SOCK_PATH) + - --v=2 + livenessProbe: + exec: + command: + - /csi-node-driver-registrar + - --kubelet-registration-path=$(DRIVER_REG_SOCK_PATH) + - --mode=kubelet-registration-probe + initialDelaySeconds: 30 + timeoutSeconds: 15 + env: + - name: ADDRESS + value: /csi/csi.sock + - name: DRIVER_REG_SOCK_PATH + value: /var/lib/kubelet/plugins/disk.csi.azure.com/csi.sock + volumeMounts: + - name: socket-dir + mountPath: /csi + - name: registration-dir + mountPath: /registration + resources: + limits: + memory: 100Mi + requests: + cpu: 10m + memory: 20Mi + - name: azuredisk + image: {{ azure_csi_plugin_image_repo }}/azuredisk-csi:{{ azure_csi_plugin_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - "--v=5" + - "--endpoint=$(CSI_ENDPOINT)" + - "--nodeid=$(KUBE_NODE_NAME)" + - "--metrics-address=0.0.0.0:29605" + - "--enable-perf-optimization=true" + - "--drivername=disk.csi.azure.com" + - "--volume-attach-limit=-1" + - "--cloud-config-secret-name=cloud-config" + - "--cloud-config-secret-namespace=kube-system" + ports: + - containerPort: 29603 + name: healthz + protocol: TCP + - containerPort: 29605 + name: metrics + protocol: TCP + livenessProbe: + failureThreshold: 5 + httpGet: + path: /healthz + port: healthz + initialDelaySeconds: 30 + timeoutSeconds: 10 + periodSeconds: 30 + env: + - name: AZURE_CREDENTIAL_FILE + value: "/etc/kubernetes/azure.json" + - name: CSI_ENDPOINT + value: unix:///csi/csi.sock + - name: KUBE_NODE_NAME + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: spec.nodeName + securityContext: + privileged: true + volumeMounts: + - mountPath: /csi + name: socket-dir + - mountPath: /var/lib/kubelet/ + mountPropagation: Bidirectional + name: mountpoint-dir + - mountPath: /etc/kubernetes/ + name: azure-cred + - mountPath: /dev + name: device-dir + - mountPath: /sys/bus/scsi/devices + name: sys-devices-dir + - mountPath: /sys/class/scsi_host/ + name: scsi-host-dir + resources: + limits: + memory: 200Mi + requests: + cpu: 10m + memory: 20Mi + volumes: + - hostPath: + path: /var/lib/kubelet/plugins/disk.csi.azure.com + type: DirectoryOrCreate + name: socket-dir + - hostPath: + path: /var/lib/kubelet/ + type: DirectoryOrCreate + name: mountpoint-dir + - hostPath: + path: /var/lib/kubelet/plugins_registry/ + type: DirectoryOrCreate + name: registration-dir + - secret: + defaultMode: 0644 + secretName: cloud-config + name: azure-cred + - hostPath: + path: /dev + type: Directory + name: device-dir + - hostPath: + path: /sys/bus/scsi/devices + type: Directory + name: sys-devices-dir + - hostPath: + path: /sys/class/scsi_host/ + type: Directory + name: scsi-host-dir diff --git a/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-cloud-config-secret.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-cloud-config-secret.yml.j2 new file mode 100644 index 0000000..f259cec --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-cloud-config-secret.yml.j2 @@ -0,0 +1,7 @@ +kind: Secret +apiVersion: v1 +metadata: + name: cloud-config + namespace: kube-system +data: + azure.json: {{ cloud_config_secret.content }} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-cloud-config.j2 b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-cloud-config.j2 new file mode 100644 index 0000000..d3932f5 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/azuredisk/templates/azure-csi-cloud-config.j2 @@ -0,0 +1,14 @@ +{ + "cloud":"AzurePublicCloud", + "tenantId": "{{ azure_csi_tenant_id }}", + "subscriptionId": "{{ azure_csi_subscription_id }}", + "aadClientId": "{{ azure_csi_aad_client_id }}", + "aadClientSecret": "{{ azure_csi_aad_client_secret }}", + "location": "{{ azure_csi_location }}", + "resourceGroup": "{{ azure_csi_resource_group }}", + "vnetName": "{{ azure_csi_vnet_name }}", + "vnetResourceGroup": "{{ azure_csi_vnet_resource_group }}", + "subnetName": "{{ azure_csi_subnet_name }}", + "securityGroupName": "{{ azure_csi_security_group_name }}", + "useInstanceMetadata": {{ azure_csi_use_instance_metadata }}, +} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/cinder/defaults/main.yml b/kubespray/roles/kubernetes-apps/csi_driver/cinder/defaults/main.yml new file mode 100644 index 0000000..501f368 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/cinder/defaults/main.yml @@ -0,0 +1,30 @@ +--- +# To access Cinder, the CSI controller will need credentials to access +# openstack apis. Per default this values will be +# read from the environment. +cinder_auth_url: "{{ lookup('env', 'OS_AUTH_URL') }}" +cinder_username: "{{ lookup('env', 'OS_USERNAME') }}" +cinder_password: "{{ lookup('env', 'OS_PASSWORD') }}" +cinder_application_credential_id: "{{ lookup('env', 'OS_APPLICATION_CREDENTIAL_ID') }}" +cinder_application_credential_name: "{{ lookup('env', 'OS_APPLICATION_CREDENTIAL_NAME') }}" +cinder_application_credential_secret: "{{ lookup('env', 'OS_APPLICATION_CREDENTIAL_SECRET') }}" +cinder_region: "{{ lookup('env', 'OS_REGION_NAME') }}" +cinder_tenant_id: "{{ lookup('env', 'OS_TENANT_ID') | default(lookup('env', 'OS_PROJECT_ID'), true) }}" +cinder_tenant_name: "{{ lookup('env', 'OS_TENANT_NAME') | default(lookup('env', 'OS_PROJECT_NAME'), true) }}" +cinder_domain_name: "{{ lookup('env', 'OS_USER_DOMAIN_NAME') }}" +cinder_domain_id: "{{ lookup('env', 'OS_USER_DOMAIN_ID') }}" +cinder_cacert: "{{ lookup('env', 'OS_CACERT') }}" + +# For now, only Cinder v3 is supported in Cinder CSI driver +cinder_blockstorage_version: "v3" +cinder_csi_controller_replicas: 1 + +# Optional. Set to true, to rescan block device and verify its size before expanding +# the filesystem. +# Not all hypervizors have a /sys/class/block/XXX/device/rescan location, therefore if +# you enable this option and your hypervizor doesn't support this, you'll get a warning +# log on resize event. It is recommended to disable this option in this case. +# Defaults to false +# cinder_csi_rescan_on_resize: true + +cinder_tolerations: [] diff --git a/kubespray/roles/kubernetes-apps/csi_driver/cinder/tasks/cinder-credential-check.yml b/kubespray/roles/kubernetes-apps/csi_driver/cinder/tasks/cinder-credential-check.yml new file mode 100644 index 0000000..d797732 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/cinder/tasks/cinder-credential-check.yml @@ -0,0 +1,59 @@ +--- +- name: Cinder CSI Driver | check cinder_auth_url value + fail: + msg: "cinder_auth_url is missing" + when: cinder_auth_url is not defined or not cinder_auth_url + +- name: Cinder CSI Driver | check cinder_username value cinder_application_credential_name value + fail: + msg: "you must either set cinder_username or cinder_application_credential_name" + when: + - cinder_username is not defined or not cinder_username + - cinder_application_credential_name is not defined or not cinder_application_credential_name + +- name: Cinder CSI Driver | check cinder_application_credential_id value + fail: + msg: "cinder_application_credential_id is missing" + when: + - cinder_application_credential_name is defined + - cinder_application_credential_name | length > 0 + - cinder_application_credential_id is not defined or not cinder_application_credential_id + +- name: Cinder CSI Driver | check cinder_application_credential_secret value + fail: + msg: "cinder_application_credential_secret is missing" + when: + - cinder_application_credential_name is defined + - cinder_application_credential_name | length > 0 + - cinder_application_credential_secret is not defined or not cinder_application_credential_secret + +- name: Cinder CSI Driver | check cinder_password value + fail: + msg: "cinder_password is missing" + when: + - cinder_username is defined + - cinder_username | length > 0 + - cinder_application_credential_name is not defined or not cinder_application_credential_name + - cinder_application_credential_secret is not defined or not cinder_application_credential_secret + - cinder_password is not defined or not cinder_password + +- name: Cinder CSI Driver | check cinder_region value + fail: + msg: "cinder_region is missing" + when: cinder_region is not defined or not cinder_region + +- name: Cinder CSI Driver | check cinder_tenant_id value + fail: + msg: "one of cinder_tenant_id or cinder_tenant_name must be specified" + when: + - cinder_tenant_id is not defined or not cinder_tenant_id + - cinder_tenant_name is not defined or not cinder_tenant_name + - cinder_application_credential_name is not defined or not cinder_application_credential_name + +- name: Cinder CSI Driver | check cinder_domain_id value + fail: + msg: "one of cinder_domain_id or cinder_domain_name must be specified" + when: + - cinder_domain_id is not defined or not cinder_domain_id + - cinder_domain_name is not defined or not cinder_domain_name + - cinder_application_credential_name is not defined or not cinder_application_credential_name diff --git a/kubespray/roles/kubernetes-apps/csi_driver/cinder/tasks/cinder-write-cacert.yml b/kubespray/roles/kubernetes-apps/csi_driver/cinder/tasks/cinder-write-cacert.yml new file mode 100644 index 0000000..c6d14a2 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/cinder/tasks/cinder-write-cacert.yml @@ -0,0 +1,11 @@ +--- +# include to workaround mitogen issue +# https://github.com/dw/mitogen/issues/663 + +- name: Cinder CSI Driver | Write cacert file + copy: + src: "{{ cinder_cacert }}" + dest: "{{ kube_config_dir }}/cinder-cacert.pem" + group: "{{ kube_cert_group }}" + mode: 0640 + delegate_to: "{{ delegate_host_to_write_cacert }}" diff --git a/kubespray/roles/kubernetes-apps/csi_driver/cinder/tasks/main.yml b/kubespray/roles/kubernetes-apps/csi_driver/cinder/tasks/main.yml new file mode 100644 index 0000000..47ce6cd --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/cinder/tasks/main.yml @@ -0,0 +1,57 @@ +--- +- name: Cinder CSI Driver | Check Cinder credentials + include_tasks: cinder-credential-check.yml + +- name: Cinder CSI Driver | Write cacert file + include_tasks: cinder-write-cacert.yml + run_once: true + loop: "{{ groups['k8s_cluster'] }}" + loop_control: + loop_var: delegate_host_to_write_cacert + when: + - inventory_hostname in groups['k8s_cluster'] + - cinder_cacert is defined + - cinder_cacert | length > 0 + +- name: Cinder CSI Driver | Write Cinder cloud-config + template: + src: "cinder-csi-cloud-config.j2" + dest: "{{ kube_config_dir }}/cinder_cloud_config" + group: "{{ kube_cert_group }}" + mode: 0640 + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Cinder CSI Driver | Get base64 cloud-config + slurp: + src: "{{ kube_config_dir }}/cinder_cloud_config" + register: cloud_config_secret + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Cinder CSI Driver | Generate Manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: cinder-csi-driver, file: cinder-csi-driver.yml} + - {name: cinder-csi-cloud-config-secret, file: cinder-csi-cloud-config-secret.yml} + - {name: cinder-csi-controllerplugin, file: cinder-csi-controllerplugin-rbac.yml} + - {name: cinder-csi-controllerplugin, file: cinder-csi-controllerplugin.yml} + - {name: cinder-csi-nodeplugin, file: cinder-csi-nodeplugin-rbac.yml} + - {name: cinder-csi-nodeplugin, file: cinder-csi-nodeplugin.yml} + - {name: cinder-csi-poddisruptionbudget, file: cinder-csi-poddisruptionbudget.yml} + register: cinder_csi_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Cinder CSI Driver | Apply Manifests + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ cinder_csi_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item.file }}" diff --git a/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-cloud-config-secret.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-cloud-config-secret.yml.j2 new file mode 100644 index 0000000..cb3cba6 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-cloud-config-secret.yml.j2 @@ -0,0 +1,10 @@ +# This YAML file contains secret objects, +# which are necessary to run csi cinder plugin. + +kind: Secret +apiVersion: v1 +metadata: + name: cloud-config + namespace: kube-system +data: + cloud.conf: {{ cloud_config_secret.content }} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-cloud-config.j2 b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-cloud-config.j2 new file mode 100644 index 0000000..04d0c68 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-cloud-config.j2 @@ -0,0 +1,44 @@ +[Global] +auth-url="{{ cinder_auth_url }}" +{% if cinder_application_credential_id|length == 0 and cinder_application_credential_name|length == 0 %} +username="{{ cinder_username }}" +password="{{ cinder_password }}" +{% endif %} +{% if cinder_application_credential_id|length > 0 %} +application-credential-id={{ cinder_application_credential_id }} +{% endif %} +{% if cinder_application_credential_name|length > 0 %} +application-credential-name={{ cinder_application_credential_name }} +{% endif %} +{% if cinder_application_credential_secret|length > 0 %} +application-credential-secret={{ cinder_application_credential_secret }} +{% endif %} +region="{{ cinder_region }}" +{% if cinder_tenant_id|length > 0 %} +tenant-id="{{ cinder_tenant_id }}" +{% endif %} +{% if cinder_tenant_name|length > 0 %} +tenant-name="{{ cinder_tenant_name }}" +{% endif %} +{% if cinder_domain_name|length > 0 %} +domain-name="{{ cinder_domain_name }}" +{% elif cinder_domain_id|length > 0 %} +domain-id ="{{ cinder_domain_id }}" +{% endif %} +{% if cinder_cacert|length > 0 %} +ca-file="{{ kube_config_dir }}/cinder-cacert.pem" +{% endif %} + +[BlockStorage] +{% if cinder_blockstorage_version is defined %} +bs-version={{ cinder_blockstorage_version }} +{% endif %} +{% if cinder_csi_ignore_volume_az is defined %} +ignore-volume-az={{ cinder_csi_ignore_volume_az | bool }} +{% endif %} +{% if node_volume_attach_limit is defined and node_volume_attach_limit != "" %} +node-volume-attach-limit="{{ node_volume_attach_limit }}" +{% endif %} +{% if cinder_csi_rescan_on_resize is defined %} +rescan-on-resize={{ cinder_csi_rescan_on_resize | bool }} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-controllerplugin-rbac.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-controllerplugin-rbac.yml.j2 new file mode 100644 index 0000000..09ecacb --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-controllerplugin-rbac.yml.j2 @@ -0,0 +1,179 @@ +# This YAML file contains RBAC API objects, +# which are necessary to run csi controller plugin + +apiVersion: v1 +kind: ServiceAccount +metadata: + name: csi-cinder-controller-sa + namespace: kube-system + +--- +# external attacher +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-attacher-role +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "patch"] + - apiGroups: ["storage.k8s.io"] + resources: ["csinodes"] + verbs: ["get", "list", "watch"] + - apiGroups: ["storage.k8s.io"] + resources: ["volumeattachments"] + verbs: ["get", "list", "watch", "patch"] + - apiGroups: ["storage.k8s.io"] + resources: ["volumeattachments/status"] + verbs: ["patch"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "watch", "list", "delete", "update", "create"] + +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-attacher-binding +subjects: + - kind: ServiceAccount + name: csi-cinder-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-attacher-role + apiGroup: rbac.authorization.k8s.io + +--- +# external Provisioner +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-provisioner-role +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "create", "delete"] + - apiGroups: [""] + resources: ["persistentvolumeclaims"] + verbs: ["get", "list", "watch", "update"] + - apiGroups: ["storage.k8s.io"] + resources: ["storageclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["nodes"] + verbs: ["get", "list", "watch"] + - apiGroups: ["storage.k8s.io"] + resources: ["csinodes"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["list", "watch", "create", "update", "patch"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshots"] + verbs: ["get", "list"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotcontents"] + verbs: ["get", "list"] + - apiGroups: ["storage.k8s.io"] + resources: ["volumeattachments"] + verbs: ["get", "list", "watch"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "watch", "list", "delete", "update", "create"] +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-provisioner-binding +subjects: + - kind: ServiceAccount + name: csi-cinder-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-provisioner-role + apiGroup: rbac.authorization.k8s.io + +--- +# external snapshotter +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-snapshotter-role +rules: + - apiGroups: [""] + resources: ["events"] + verbs: ["list", "watch", "create", "update", "patch"] + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotcontents"] + verbs: ["create", "get", "list", "watch", "update", "patch", "delete"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotcontents/status"] + verbs: ["update"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "watch", "list", "delete", "update", "create"] +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-snapshotter-binding +subjects: + - kind: ServiceAccount + name: csi-cinder-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-snapshotter-role + apiGroup: rbac.authorization.k8s.io +--- + +# External Resizer +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-resizer-role +rules: + # The following rule should be uncommented for plugins that require secrets + # for provisioning. + # - apiGroups: [""] + # resources: ["secrets"] + # verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "patch"] + - apiGroups: [""] + resources: ["persistentvolumeclaims"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["pods"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["persistentvolumeclaims/status"] + verbs: ["patch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["list", "watch", "create", "update", "patch"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "watch", "list", "delete", "update", "create"] +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-resizer-binding +subjects: + - kind: ServiceAccount + name: csi-cinder-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-resizer-role + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-controllerplugin.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-controllerplugin.yml.j2 new file mode 100644 index 0000000..4fe7e47 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-controllerplugin.yml.j2 @@ -0,0 +1,171 @@ +# This YAML file contains CSI Controller Plugin Sidecars +# external-attacher, external-provisioner, external-snapshotter + +--- +kind: Deployment +apiVersion: apps/v1 +metadata: + name: csi-cinder-controllerplugin + namespace: kube-system +spec: + replicas: {{ cinder_csi_controller_replicas }} + selector: + matchLabels: + app: csi-cinder-controllerplugin + template: + metadata: + labels: + app: csi-cinder-controllerplugin + spec: + serviceAccountName: csi-cinder-controller-sa + containers: + - name: csi-attacher + image: {{ csi_attacher_image_repo }}:{{ csi_attacher_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - "--csi-address=$(ADDRESS)" + - "--timeout=3m" +{% if cinder_csi_controller_replicas is defined and cinder_csi_controller_replicas > 1 %} + - --leader-election=true +{% endif %} + env: + - name: ADDRESS + value: /var/lib/csi/sockets/pluginproxy/csi.sock + volumeMounts: + - name: socket-dir + mountPath: /var/lib/csi/sockets/pluginproxy/ + - name: csi-provisioner + image: {{ csi_provisioner_image_repo }}:{{ csi_provisioner_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - "--csi-address=$(ADDRESS)" + - "--timeout=3m" + - "--default-fstype=ext4" + - "--extra-create-metadata" +{% if cinder_topology is defined and cinder_topology %} + - --feature-gates=Topology=true +{% endif %} +{% if cinder_csi_controller_replicas is defined and cinder_csi_controller_replicas > 1 %} + - "--leader-election=true" +{% endif %} + env: + - name: ADDRESS + value: /var/lib/csi/sockets/pluginproxy/csi.sock + volumeMounts: + - name: socket-dir + mountPath: /var/lib/csi/sockets/pluginproxy/ + - name: csi-snapshotter + image: {{ csi_snapshotter_image_repo }}:{{ csi_snapshotter_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - "--csi-address=$(ADDRESS)" + - "--timeout=3m" + - "--extra-create-metadata" +{% if cinder_csi_controller_replicas is defined and cinder_csi_controller_replicas > 1 %} + - --leader-election=true +{% endif %} + env: + - name: ADDRESS + value: /var/lib/csi/sockets/pluginproxy/csi.sock + volumeMounts: + - mountPath: /var/lib/csi/sockets/pluginproxy/ + name: socket-dir + - name: csi-resizer + image: {{ csi_resizer_image_repo }}:{{ csi_resizer_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - "--csi-address=$(ADDRESS)" + - "--timeout=3m" + - "--handle-volume-inuse-error=false" +{% if cinder_csi_controller_replicas is defined and cinder_csi_controller_replicas > 1 %} + - --leader-election=true +{% endif %} + env: + - name: ADDRESS + value: /var/lib/csi/sockets/pluginproxy/csi.sock + volumeMounts: + - name: socket-dir + mountPath: /var/lib/csi/sockets/pluginproxy/ + - name: liveness-probe + image: {{ csi_livenessprobe_image_repo }}:{{ csi_livenessprobe_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - "--csi-address=$(ADDRESS)" + env: + - name: ADDRESS + value: /var/lib/csi/sockets/pluginproxy/csi.sock + volumeMounts: + - mountPath: /var/lib/csi/sockets/pluginproxy/ + name: socket-dir + - name: cinder-csi-plugin + image: {{ cinder_csi_plugin_image_repo }}:{{ cinder_csi_plugin_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - /bin/cinder-csi-plugin + - "--endpoint=$(CSI_ENDPOINT)" + - "--cloud-config=$(CLOUD_CONFIG)" + - "--cluster=$(CLUSTER_NAME)" + env: + - name: CSI_ENDPOINT + value: unix://csi/csi.sock + - name: CLOUD_CONFIG + value: /etc/config/cloud.conf + - name: CLUSTER_NAME + value: {{ cluster_name }} + ports: + - containerPort: 9808 + name: healthz + protocol: TCP + livenessProbe: + failureThreshold: 5 + httpGet: + path: /healthz + port: healthz + initialDelaySeconds: 10 + timeoutSeconds: 10 + periodSeconds: 60 + volumeMounts: + - name: socket-dir + mountPath: /csi + - name: secret-cinderplugin + mountPath: /etc/config + readOnly: true + - name: ca-certs + mountPath: /etc/ssl/certs + readOnly: true +{% if ssl_ca_dirs | length %} +{% for dir in ssl_ca_dirs %} + - name: {{ dir | regex_replace('^/(.*)$', '\\1' ) | regex_replace('/', '-') }} + mountPath: {{ dir }} + readOnly: true +{% endfor %} +{% endif %} +{% if cinder_cacert is defined and cinder_cacert != "" %} + - name: cinder-cacert + mountPath: {{ kube_config_dir }}/cinder-cacert.pem + readOnly: true +{% endif %} + volumes: + - name: socket-dir + emptyDir: + - name: secret-cinderplugin + secret: + secretName: cloud-config + - name: ca-certs + hostPath: + path: /etc/ssl/certs + type: DirectoryOrCreate +{% if ssl_ca_dirs | length %} +{% for dir in ssl_ca_dirs %} + - name: {{ dir | regex_replace('^/(.*)$', '\\1' ) | regex_replace('/', '-') }} + hostPath: + path: {{ dir }} + type: DirectoryOrCreate +{% endfor %} +{% endif %} +{% if cinder_cacert is defined and cinder_cacert != "" %} + - name: cinder-cacert + hostPath: + path: {{ kube_config_dir }}/cinder-cacert.pem + type: FileOrCreate +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-driver.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-driver.yml.j2 new file mode 100644 index 0000000..5b681e4 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-driver.yml.j2 @@ -0,0 +1,10 @@ +apiVersion: storage.k8s.io/v1 +kind: CSIDriver +metadata: + name: cinder.csi.openstack.org +spec: + attachRequired: true + podInfoOnMount: true + volumeLifecycleModes: + - Persistent + - Ephemeral diff --git a/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-nodeplugin-rbac.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-nodeplugin-rbac.yml.j2 new file mode 100644 index 0000000..db58963 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-nodeplugin-rbac.yml.j2 @@ -0,0 +1,38 @@ +# This YAML defines all API objects to create RBAC roles for csi node plugin. + +apiVersion: v1 +kind: ServiceAccount +metadata: + name: csi-cinder-node-sa + namespace: kube-system +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-nodeplugin-role +rules: + - apiGroups: [""] + resources: ["events"] + verbs: ["get", "list", "watch", "create", "update", "patch"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotcontents"] + verbs: ["get", "list", "watch"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotcontents/status"] + verbs: ["update"] +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-nodeplugin-binding +subjects: + - kind: ServiceAccount + name: csi-cinder-node-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-nodeplugin-role + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-nodeplugin.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-nodeplugin.yml.j2 new file mode 100644 index 0000000..289b168 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-nodeplugin.yml.j2 @@ -0,0 +1,145 @@ +# This YAML file contains driver-registrar & csi driver nodeplugin API objects, +# which are necessary to run csi nodeplugin for cinder. + +kind: DaemonSet +apiVersion: apps/v1 +metadata: + name: csi-cinder-nodeplugin + namespace: kube-system +spec: + selector: + matchLabels: + app: csi-cinder-nodeplugin + template: + metadata: + labels: + app: csi-cinder-nodeplugin + spec: + serviceAccountName: csi-cinder-node-sa + hostNetwork: true + containers: + - name: node-driver-registrar + image: {{ csi_node_driver_registrar_image_repo }}:{{ csi_node_driver_registrar_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - "--csi-address=$(ADDRESS)" + - "--kubelet-registration-path=$(DRIVER_REG_SOCK_PATH)" + env: + - name: ADDRESS + value: /csi/csi.sock + - name: DRIVER_REG_SOCK_PATH + value: /var/lib/kubelet/plugins/cinder.csi.openstack.org/csi.sock + - name: KUBE_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + volumeMounts: + - name: socket-dir + mountPath: /csi + - name: registration-dir + mountPath: /registration + - name: liveness-probe + image: {{ csi_livenessprobe_image_repo }}:{{ csi_livenessprobe_image_tag }} + args: + - "--csi-address=/csi/csi.sock" + volumeMounts: + - name: socket-dir + mountPath: /csi + - name: cinder-csi-plugin + securityContext: + privileged: true + capabilities: + add: ["SYS_ADMIN"] + allowPrivilegeEscalation: true + image: {{ cinder_csi_plugin_image_repo }}:{{ cinder_csi_plugin_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - /bin/cinder-csi-plugin + - "--endpoint=$(CSI_ENDPOINT)" + - "--cloud-config=$(CLOUD_CONFIG)" + env: + - name: CSI_ENDPOINT + value: unix://csi/csi.sock + - name: CLOUD_CONFIG + value: /etc/config/cloud.conf + ports: + - containerPort: 9808 + name: healthz + protocol: TCP + livenessProbe: + failureThreshold: 5 + httpGet: + path: /healthz + port: healthz + initialDelaySeconds: 10 + timeoutSeconds: 3 + periodSeconds: 10 + volumeMounts: + - name: socket-dir + mountPath: /csi + - name: kubelet-dir + mountPath: /var/lib/kubelet + mountPropagation: "Bidirectional" + - name: pods-probe-dir + mountPath: /dev + mountPropagation: "HostToContainer" + - name: secret-cinderplugin + mountPath: /etc/config + readOnly: true + - name: ca-certs + mountPath: /etc/ssl/certs + readOnly: true +{% if ssl_ca_dirs | length %} +{% for dir in ssl_ca_dirs %} + - name: {{ dir | regex_replace('^/(.*)$', '\\1' ) | regex_replace('/', '-') }} + mountPath: {{ dir }} + readOnly: true +{% endfor %} +{% endif %} +{% if cinder_cacert is defined and cinder_cacert != "" %} + - name: cinder-cacert + mountPath: {{ kube_config_dir }}/cinder-cacert.pem + readOnly: true +{% endif %} + volumes: + - name: socket-dir + hostPath: + path: /var/lib/kubelet/plugins/cinder.csi.openstack.org + type: DirectoryOrCreate + - name: registration-dir + hostPath: + path: /var/lib/kubelet/plugins_registry/ + type: Directory + - name: kubelet-dir + hostPath: + path: /var/lib/kubelet + type: Directory + - name: pods-probe-dir + hostPath: + path: /dev + type: Directory + - name: secret-cinderplugin + secret: + secretName: cloud-config + - name: ca-certs + hostPath: + path: /etc/ssl/certs + type: DirectoryOrCreate +{% if ssl_ca_dirs | length %} +{% for dir in ssl_ca_dirs %} + - name: {{ dir | regex_replace('^/(.*)$', '\\1' ) | regex_replace('/', '-') }} + hostPath: + path: {{ dir }} + type: DirectoryOrCreate +{% endfor %} +{% endif %} +{% if cinder_cacert is defined and cinder_cacert != "" %} + - name: cinder-cacert + hostPath: + path: {{ kube_config_dir }}/cinder-cacert.pem + type: FileOrCreate +{% endif %} +{% if cinder_tolerations %} + tolerations: + {{ cinder_tolerations | to_nice_yaml(indent=2) | indent(width=8) }} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-poddisruptionbudget.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-poddisruptionbudget.yml.j2 new file mode 100644 index 0000000..391d3b3 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/cinder/templates/cinder-csi-poddisruptionbudget.yml.j2 @@ -0,0 +1,14 @@ +apiVersion: policy/v1 +kind: PodDisruptionBudget +metadata: + name: cinder-csi-pdb + namespace: kube-system +spec: +{% if cinder_csi_controller_replicas is defined and cinder_csi_controller_replicas > 1 %} + minAvailable: 1 +{% else %} + minAvailable: 0 +{% endif %} + selector: + matchLabels: + app: csi-cinder-controllerplugin diff --git a/kubespray/roles/kubernetes-apps/csi_driver/csi_crd/tasks/main.yml b/kubespray/roles/kubernetes-apps/csi_driver/csi_crd/tasks/main.yml new file mode 100644 index 0000000..4790931 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/csi_crd/tasks/main.yml @@ -0,0 +1,26 @@ +--- +- name: CSI CRD | Generate Manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: volumesnapshotclasses, file: volumesnapshotclasses.yml} + - {name: volumesnapshotcontents, file: volumesnapshotcontents.yml} + - {name: volumesnapshots, file: volumesnapshots.yml} + register: csi_crd_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: CSI CRD | Apply Manifests + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + wait: true + with_items: + - "{{ csi_crd_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item.file }}" diff --git a/kubespray/roles/kubernetes-apps/csi_driver/csi_crd/templates/volumesnapshotclasses.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/csi_crd/templates/volumesnapshotclasses.yml.j2 new file mode 100644 index 0000000..47e5fd3 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/csi_crd/templates/volumesnapshotclasses.yml.j2 @@ -0,0 +1,116 @@ + +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.4.0 + api-approved.kubernetes.io: "https://github.com/kubernetes-csi/external-snapshotter/pull/419" + creationTimestamp: null + name: volumesnapshotclasses.snapshot.storage.k8s.io +spec: + group: snapshot.storage.k8s.io + names: + kind: VolumeSnapshotClass + listKind: VolumeSnapshotClassList + plural: volumesnapshotclasses + singular: volumesnapshotclass + scope: Cluster + versions: + - additionalPrinterColumns: + - jsonPath: .driver + name: Driver + type: string + - description: Determines whether a VolumeSnapshotContent created through the VolumeSnapshotClass should be deleted when its bound VolumeSnapshot is deleted. + jsonPath: .deletionPolicy + name: DeletionPolicy + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1 + schema: + openAPIV3Schema: + description: VolumeSnapshotClass specifies parameters that a underlying storage system uses when creating a volume snapshot. A specific VolumeSnapshotClass is used by specifying its name in a VolumeSnapshot object. VolumeSnapshotClasses are non-namespaced + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation of an object. Servers should convert recognized schemas to the latest internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + deletionPolicy: + description: deletionPolicy determines whether a VolumeSnapshotContent created through the VolumeSnapshotClass should be deleted when its bound VolumeSnapshot is deleted. Supported values are "Retain" and "Delete". "Retain" means that the VolumeSnapshotContent and its physical snapshot on underlying storage system are kept. "Delete" means that the VolumeSnapshotContent and its physical snapshot on underlying storage system are deleted. Required. + enum: + - Delete + - Retain + type: string + driver: + description: driver is the name of the storage driver that handles this VolumeSnapshotClass. Required. + type: string + kind: + description: 'Kind is a string value representing the REST resource this object represents. Servers may infer this from the endpoint the client submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + parameters: + additionalProperties: + type: string + description: parameters is a key-value map with storage driver specific parameters for creating snapshots. These values are opaque to Kubernetes. + type: object + required: + - deletionPolicy + - driver + type: object + served: true + storage: true + subresources: {} + - additionalPrinterColumns: + - jsonPath: .driver + name: Driver + type: string + - description: Determines whether a VolumeSnapshotContent created through the VolumeSnapshotClass should be deleted when its bound VolumeSnapshot is deleted. + jsonPath: .deletionPolicy + name: DeletionPolicy + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1beta1 + # This indicates the v1beta1 version of the custom resource is deprecated. + # API requests to this version receive a warning in the server response. + deprecated: true + # This overrides the default warning returned to clients making v1beta1 API requests. + deprecationWarning: "snapshot.storage.k8s.io/v1beta1 VolumeSnapshotClass is deprecated; use snapshot.storage.k8s.io/v1 VolumeSnapshotClass" + schema: + openAPIV3Schema: + description: VolumeSnapshotClass specifies parameters that a underlying storage system uses when creating a volume snapshot. A specific VolumeSnapshotClass is used by specifying its name in a VolumeSnapshot object. VolumeSnapshotClasses are non-namespaced + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation of an object. Servers should convert recognized schemas to the latest internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + deletionPolicy: + description: deletionPolicy determines whether a VolumeSnapshotContent created through the VolumeSnapshotClass should be deleted when its bound VolumeSnapshot is deleted. Supported values are "Retain" and "Delete". "Retain" means that the VolumeSnapshotContent and its physical snapshot on underlying storage system are kept. "Delete" means that the VolumeSnapshotContent and its physical snapshot on underlying storage system are deleted. Required. + enum: + - Delete + - Retain + type: string + driver: + description: driver is the name of the storage driver that handles this VolumeSnapshotClass. Required. + type: string + kind: + description: 'Kind is a string value representing the REST resource this object represents. Servers may infer this from the endpoint the client submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + parameters: + additionalProperties: + type: string + description: parameters is a key-value map with storage driver specific parameters for creating snapshots. These values are opaque to Kubernetes. + type: object + required: + - deletionPolicy + - driver + type: object + served: true + storage: false + subresources: {} +status: + acceptedNames: + kind: "" + plural: "" + conditions: [] + storedVersions: [] diff --git a/kubespray/roles/kubernetes-apps/csi_driver/csi_crd/templates/volumesnapshotcontents.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/csi_crd/templates/volumesnapshotcontents.yml.j2 new file mode 100644 index 0000000..c611221 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/csi_crd/templates/volumesnapshotcontents.yml.j2 @@ -0,0 +1,305 @@ + +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.4.0 + api-approved.kubernetes.io: "https://github.com/kubernetes-csi/external-snapshotter/pull/419" + creationTimestamp: null + name: volumesnapshotcontents.snapshot.storage.k8s.io +spec: + group: snapshot.storage.k8s.io + names: + kind: VolumeSnapshotContent + listKind: VolumeSnapshotContentList + plural: volumesnapshotcontents + singular: volumesnapshotcontent + scope: Cluster + versions: + - additionalPrinterColumns: + - description: Indicates if the snapshot is ready to be used to restore a volume. + jsonPath: .status.readyToUse + name: ReadyToUse + type: boolean + - description: Represents the complete size of the snapshot in bytes + jsonPath: .status.restoreSize + name: RestoreSize + type: integer + - description: Determines whether this VolumeSnapshotContent and its physical snapshot on the underlying storage system should be deleted when its bound VolumeSnapshot is deleted. + jsonPath: .spec.deletionPolicy + name: DeletionPolicy + type: string + - description: Name of the CSI driver used to create the physical snapshot on the underlying storage system. + jsonPath: .spec.driver + name: Driver + type: string + - description: Name of the VolumeSnapshotClass to which this snapshot belongs. + jsonPath: .spec.volumeSnapshotClassName + name: VolumeSnapshotClass + type: string + - description: Name of the VolumeSnapshot object to which this VolumeSnapshotContent object is bound. + jsonPath: .spec.volumeSnapshotRef.name + name: VolumeSnapshot + type: string + - description: Namespace of the VolumeSnapshot object to which this VolumeSnapshotContent object is bound. + jsonPath: .spec.volumeSnapshotRef.namespace + name: VolumeSnapshotNamespace + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1 + schema: + openAPIV3Schema: + description: VolumeSnapshotContent represents the actual "on-disk" snapshot object in the underlying storage system + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation of an object. Servers should convert recognized schemas to the latest internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this object represents. Servers may infer this from the endpoint the client submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + spec: + description: spec defines properties of a VolumeSnapshotContent created by the underlying storage system. Required. + properties: + deletionPolicy: + description: deletionPolicy determines whether this VolumeSnapshotContent and its physical snapshot on the underlying storage system should be deleted when its bound VolumeSnapshot is deleted. Supported values are "Retain" and "Delete". "Retain" means that the VolumeSnapshotContent and its physical snapshot on underlying storage system are kept. "Delete" means that the VolumeSnapshotContent and its physical snapshot on underlying storage system are deleted. For dynamically provisioned snapshots, this field will automatically be filled in by the CSI snapshotter sidecar with the "DeletionPolicy" field defined in the corresponding VolumeSnapshotClass. For pre-existing snapshots, users MUST specify this field when creating the VolumeSnapshotContent object. Required. + enum: + - Delete + - Retain + type: string + driver: + description: driver is the name of the CSI driver used to create the physical snapshot on the underlying storage system. This MUST be the same as the name returned by the CSI GetPluginName() call for that driver. Required. + type: string + source: + description: source specifies whether the snapshot is (or should be) dynamically provisioned or already exists, and just requires a Kubernetes object representation. This field is immutable after creation. Required. + properties: + snapshotHandle: + description: snapshotHandle specifies the CSI "snapshot_id" of a pre-existing snapshot on the underlying storage system for which a Kubernetes object representation was (or should be) created. This field is immutable. + type: string + volumeHandle: + description: volumeHandle specifies the CSI "volume_id" of the volume from which a snapshot should be dynamically taken from. This field is immutable. + type: string + type: object + oneOf: + - required: ["snapshotHandle"] + - required: ["volumeHandle"] + volumeSnapshotClassName: + description: name of the VolumeSnapshotClass from which this snapshot was (or will be) created. Note that after provisioning, the VolumeSnapshotClass may be deleted or recreated with different set of values, and as such, should not be referenced post-snapshot creation. + type: string + volumeSnapshotRef: + description: volumeSnapshotRef specifies the VolumeSnapshot object to which this VolumeSnapshotContent object is bound. VolumeSnapshot.Spec.VolumeSnapshotContentName field must reference to this VolumeSnapshotContent's name for the bidirectional binding to be valid. For a pre-existing VolumeSnapshotContent object, name and namespace of the VolumeSnapshot object MUST be provided for binding to happen. This field is immutable after creation. Required. + properties: + apiVersion: + description: API version of the referent. + type: string + fieldPath: + description: 'If referring to a piece of an object instead of an entire object, this string should contain a valid JSON/Go field access statement, such as desiredState.manifest.containers[2]. For example, if the object reference is to a container within a pod, this would take on a value like: "spec.containers{name}" (where "name" refers to the name of the container that triggered the event) or if no container name is specified "spec.containers[2]" (container with index 2 in this pod). This syntax is chosen only to have some well-defined way of referencing a part of an object. TODO: this design is not final and this field is subject to change in the future.' + type: string + kind: + description: 'Kind of the referent. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + namespace: + description: 'Namespace of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/' + type: string + resourceVersion: + description: 'Specific resourceVersion to which this reference is made, if any. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency' + type: string + uid: + description: 'UID of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids' + type: string + type: object + required: + - deletionPolicy + - driver + - source + - volumeSnapshotRef + type: object + status: + description: status represents the current information of a snapshot. + properties: + creationTime: + description: creationTime is the timestamp when the point-in-time snapshot is taken by the underlying storage system. In dynamic snapshot creation case, this field will be filled in by the CSI snapshotter sidecar with the "creation_time" value returned from CSI "CreateSnapshot" gRPC call. For a pre-existing snapshot, this field will be filled with the "creation_time" value returned from the CSI "ListSnapshots" gRPC call if the driver supports it. If not specified, it indicates the creation time is unknown. The format of this field is a Unix nanoseconds time encoded as an int64. On Unix, the command `date +%s%N` returns the current time in nanoseconds since 1970-01-01 00:00:00 UTC. + format: int64 + type: integer + error: + description: error is the last observed error during snapshot creation, if any. Upon success after retry, this error field will be cleared. + properties: + message: + description: 'message is a string detailing the encountered error during snapshot creation if specified. NOTE: message may be logged, and it should not contain sensitive information.' + type: string + time: + description: time is the timestamp when the error was encountered. + format: date-time + type: string + type: object + readyToUse: + description: readyToUse indicates if a snapshot is ready to be used to restore a volume. In dynamic snapshot creation case, this field will be filled in by the CSI snapshotter sidecar with the "ready_to_use" value returned from CSI "CreateSnapshot" gRPC call. For a pre-existing snapshot, this field will be filled with the "ready_to_use" value returned from the CSI "ListSnapshots" gRPC call if the driver supports it, otherwise, this field will be set to "True". If not specified, it means the readiness of a snapshot is unknown. + type: boolean + restoreSize: + description: restoreSize represents the complete size of the snapshot in bytes. In dynamic snapshot creation case, this field will be filled in by the CSI snapshotter sidecar with the "size_bytes" value returned from CSI "CreateSnapshot" gRPC call. For a pre-existing snapshot, this field will be filled with the "size_bytes" value returned from the CSI "ListSnapshots" gRPC call if the driver supports it. When restoring a volume from this snapshot, the size of the volume MUST NOT be smaller than the restoreSize if it is specified, otherwise the restoration will fail. If not specified, it indicates that the size is unknown. + format: int64 + minimum: 0 + type: integer + snapshotHandle: + description: snapshotHandle is the CSI "snapshot_id" of a snapshot on the underlying storage system. If not specified, it indicates that dynamic snapshot creation has either failed or it is still in progress. + type: string + type: object + required: + - spec + type: object + served: true + storage: true + subresources: + status: {} + - additionalPrinterColumns: + - description: Indicates if the snapshot is ready to be used to restore a volume. + jsonPath: .status.readyToUse + name: ReadyToUse + type: boolean + - description: Represents the complete size of the snapshot in bytes + jsonPath: .status.restoreSize + name: RestoreSize + type: integer + - description: Determines whether this VolumeSnapshotContent and its physical snapshot on the underlying storage system should be deleted when its bound VolumeSnapshot is deleted. + jsonPath: .spec.deletionPolicy + name: DeletionPolicy + type: string + - description: Name of the CSI driver used to create the physical snapshot on the underlying storage system. + jsonPath: .spec.driver + name: Driver + type: string + - description: Name of the VolumeSnapshotClass to which this snapshot belongs. + jsonPath: .spec.volumeSnapshotClassName + name: VolumeSnapshotClass + type: string + - description: Name of the VolumeSnapshot object to which this VolumeSnapshotContent object is bound. + jsonPath: .spec.volumeSnapshotRef.name + name: VolumeSnapshot + type: string + - description: Namespace of the VolumeSnapshot object to which this VolumeSnapshotContent object is bound. + jsonPath: .spec.volumeSnapshotRef.namespace + name: VolumeSnapshotNamespace + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1beta1 + # This indicates the v1beta1 version of the custom resource is deprecated. + # API requests to this version receive a warning in the server response. + deprecated: true + # This overrides the default warning returned to clients making v1beta1 API requests. + deprecationWarning: "snapshot.storage.k8s.io/v1beta1 VolumeSnapshotContent is deprecated; use snapshot.storage.k8s.io/v1 VolumeSnapshotContent" + schema: + openAPIV3Schema: + description: VolumeSnapshotContent represents the actual "on-disk" snapshot object in the underlying storage system + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation of an object. Servers should convert recognized schemas to the latest internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this object represents. Servers may infer this from the endpoint the client submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + spec: + description: spec defines properties of a VolumeSnapshotContent created by the underlying storage system. Required. + properties: + deletionPolicy: + description: deletionPolicy determines whether this VolumeSnapshotContent and its physical snapshot on the underlying storage system should be deleted when its bound VolumeSnapshot is deleted. Supported values are "Retain" and "Delete". "Retain" means that the VolumeSnapshotContent and its physical snapshot on underlying storage system are kept. "Delete" means that the VolumeSnapshotContent and its physical snapshot on underlying storage system are deleted. For dynamically provisioned snapshots, this field will automatically be filled in by the CSI snapshotter sidecar with the "DeletionPolicy" field defined in the corresponding VolumeSnapshotClass. For pre-existing snapshots, users MUST specify this field when creating the VolumeSnapshotContent object. Required. + enum: + - Delete + - Retain + type: string + driver: + description: driver is the name of the CSI driver used to create the physical snapshot on the underlying storage system. This MUST be the same as the name returned by the CSI GetPluginName() call for that driver. Required. + type: string + source: + description: source specifies whether the snapshot is (or should be) dynamically provisioned or already exists, and just requires a Kubernetes object representation. This field is immutable after creation. Required. + properties: + snapshotHandle: + description: snapshotHandle specifies the CSI "snapshot_id" of a pre-existing snapshot on the underlying storage system for which a Kubernetes object representation was (or should be) created. This field is immutable. + type: string + volumeHandle: + description: volumeHandle specifies the CSI "volume_id" of the volume from which a snapshot should be dynamically taken from. This field is immutable. + type: string + type: object + volumeSnapshotClassName: + description: name of the VolumeSnapshotClass from which this snapshot was (or will be) created. Note that after provisioning, the VolumeSnapshotClass may be deleted or recreated with different set of values, and as such, should not be referenced post-snapshot creation. + type: string + volumeSnapshotRef: + description: volumeSnapshotRef specifies the VolumeSnapshot object to which this VolumeSnapshotContent object is bound. VolumeSnapshot.Spec.VolumeSnapshotContentName field must reference to this VolumeSnapshotContent's name for the bidirectional binding to be valid. For a pre-existing VolumeSnapshotContent object, name and namespace of the VolumeSnapshot object MUST be provided for binding to happen. This field is immutable after creation. Required. + properties: + apiVersion: + description: API version of the referent. + type: string + fieldPath: + description: 'If referring to a piece of an object instead of an entire object, this string should contain a valid JSON/Go field access statement, such as desiredState.manifest.containers[2]. For example, if the object reference is to a container within a pod, this would take on a value like: "spec.containers{name}" (where "name" refers to the name of the container that triggered the event) or if no container name is specified "spec.containers[2]" (container with index 2 in this pod). This syntax is chosen only to have some well-defined way of referencing a part of an object. TODO: this design is not final and this field is subject to change in the future.' + type: string + kind: + description: 'Kind of the referent. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + namespace: + description: 'Namespace of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/' + type: string + resourceVersion: + description: 'Specific resourceVersion to which this reference is made, if any. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency' + type: string + uid: + description: 'UID of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids' + type: string + type: object + required: + - deletionPolicy + - driver + - source + - volumeSnapshotRef + type: object + status: + description: status represents the current information of a snapshot. + properties: + creationTime: + description: creationTime is the timestamp when the point-in-time snapshot is taken by the underlying storage system. In dynamic snapshot creation case, this field will be filled in by the CSI snapshotter sidecar with the "creation_time" value returned from CSI "CreateSnapshot" gRPC call. For a pre-existing snapshot, this field will be filled with the "creation_time" value returned from the CSI "ListSnapshots" gRPC call if the driver supports it. If not specified, it indicates the creation time is unknown. The format of this field is a Unix nanoseconds time encoded as an int64. On Unix, the command `date +%s%N` returns the current time in nanoseconds since 1970-01-01 00:00:00 UTC. + format: int64 + type: integer + error: + description: error is the last observed error during snapshot creation, if any. Upon success after retry, this error field will be cleared. + properties: + message: + description: 'message is a string detailing the encountered error during snapshot creation if specified. NOTE: message may be logged, and it should not contain sensitive information.' + type: string + time: + description: time is the timestamp when the error was encountered. + format: date-time + type: string + type: object + readyToUse: + description: readyToUse indicates if a snapshot is ready to be used to restore a volume. In dynamic snapshot creation case, this field will be filled in by the CSI snapshotter sidecar with the "ready_to_use" value returned from CSI "CreateSnapshot" gRPC call. For a pre-existing snapshot, this field will be filled with the "ready_to_use" value returned from the CSI "ListSnapshots" gRPC call if the driver supports it, otherwise, this field will be set to "True". If not specified, it means the readiness of a snapshot is unknown. + type: boolean + restoreSize: + description: restoreSize represents the complete size of the snapshot in bytes. In dynamic snapshot creation case, this field will be filled in by the CSI snapshotter sidecar with the "size_bytes" value returned from CSI "CreateSnapshot" gRPC call. For a pre-existing snapshot, this field will be filled with the "size_bytes" value returned from the CSI "ListSnapshots" gRPC call if the driver supports it. When restoring a volume from this snapshot, the size of the volume MUST NOT be smaller than the restoreSize if it is specified, otherwise the restoration will fail. If not specified, it indicates that the size is unknown. + format: int64 + minimum: 0 + type: integer + snapshotHandle: + description: snapshotHandle is the CSI "snapshot_id" of a snapshot on the underlying storage system. If not specified, it indicates that dynamic snapshot creation has either failed or it is still in progress. + type: string + type: object + required: + - spec + type: object + served: true + storage: false + subresources: + status: {} +status: + acceptedNames: + kind: "" + plural: "" + conditions: [] + storedVersions: [] diff --git a/kubespray/roles/kubernetes-apps/csi_driver/csi_crd/templates/volumesnapshots.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/csi_crd/templates/volumesnapshots.yml.j2 new file mode 100644 index 0000000..1b41ff8 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/csi_crd/templates/volumesnapshots.yml.j2 @@ -0,0 +1,231 @@ + +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.4.0 + api-approved.kubernetes.io: "https://github.com/kubernetes-csi/external-snapshotter/pull/419" + creationTimestamp: null + name: volumesnapshots.snapshot.storage.k8s.io +spec: + group: snapshot.storage.k8s.io + names: + kind: VolumeSnapshot + listKind: VolumeSnapshotList + plural: volumesnapshots + singular: volumesnapshot + scope: Namespaced + versions: + - additionalPrinterColumns: + - description: Indicates if the snapshot is ready to be used to restore a volume. + jsonPath: .status.readyToUse + name: ReadyToUse + type: boolean + - description: If a new snapshot needs to be created, this contains the name of the source PVC from which this snapshot was (or will be) created. + jsonPath: .spec.source.persistentVolumeClaimName + name: SourcePVC + type: string + - description: If a snapshot already exists, this contains the name of the existing VolumeSnapshotContent object representing the existing snapshot. + jsonPath: .spec.source.volumeSnapshotContentName + name: SourceSnapshotContent + type: string + - description: Represents the minimum size of volume required to rehydrate from this snapshot. + jsonPath: .status.restoreSize + name: RestoreSize + type: string + - description: The name of the VolumeSnapshotClass requested by the VolumeSnapshot. + jsonPath: .spec.volumeSnapshotClassName + name: SnapshotClass + type: string + - description: Name of the VolumeSnapshotContent object to which the VolumeSnapshot object intends to bind to. Please note that verification of binding actually requires checking both VolumeSnapshot and VolumeSnapshotContent to ensure both are pointing at each other. Binding MUST be verified prior to usage of this object. + jsonPath: .status.boundVolumeSnapshotContentName + name: SnapshotContent + type: string + - description: Timestamp when the point-in-time snapshot was taken by the underlying storage system. + jsonPath: .status.creationTime + name: CreationTime + type: date + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1 + schema: + openAPIV3Schema: + description: VolumeSnapshot is a user's request for either creating a point-in-time snapshot of a persistent volume, or binding to a pre-existing snapshot. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation of an object. Servers should convert recognized schemas to the latest internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this object represents. Servers may infer this from the endpoint the client submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + spec: + description: 'spec defines the desired characteristics of a snapshot requested by a user. More info: https://kubernetes.io/docs/concepts/storage/volume-snapshots#volumesnapshots Required.' + properties: + source: + description: source specifies where a snapshot will be created from. This field is immutable after creation. Required. + properties: + persistentVolumeClaimName: + description: persistentVolumeClaimName specifies the name of the PersistentVolumeClaim object representing the volume from which a snapshot should be created. This PVC is assumed to be in the same namespace as the VolumeSnapshot object. This field should be set if the snapshot does not exists, and needs to be created. This field is immutable. + type: string + volumeSnapshotContentName: + description: volumeSnapshotContentName specifies the name of a pre-existing VolumeSnapshotContent object representing an existing volume snapshot. This field should be set if the snapshot already exists and only needs a representation in Kubernetes. This field is immutable. + type: string + type: object + oneOf: + - required: ["persistentVolumeClaimName"] + - required: ["volumeSnapshotContentName"] + volumeSnapshotClassName: + description: 'VolumeSnapshotClassName is the name of the VolumeSnapshotClass requested by the VolumeSnapshot. VolumeSnapshotClassName may be left nil to indicate that the default SnapshotClass should be used. A given cluster may have multiple default Volume SnapshotClasses: one default per CSI Driver. If a VolumeSnapshot does not specify a SnapshotClass, VolumeSnapshotSource will be checked to figure out what the associated CSI Driver is, and the default VolumeSnapshotClass associated with that CSI Driver will be used. If more than one VolumeSnapshotClass exist for a given CSI Driver and more than one have been marked as default, CreateSnapshot will fail and generate an event. Empty string is not allowed for this field.' + type: string + required: + - source + type: object + status: + description: status represents the current information of a snapshot. Consumers must verify binding between VolumeSnapshot and VolumeSnapshotContent objects is successful (by validating that both VolumeSnapshot and VolumeSnapshotContent point at each other) before using this object. + properties: + boundVolumeSnapshotContentName: + description: 'boundVolumeSnapshotContentName is the name of the VolumeSnapshotContent object to which this VolumeSnapshot object intends to bind to. If not specified, it indicates that the VolumeSnapshot object has not been successfully bound to a VolumeSnapshotContent object yet. NOTE: To avoid possible security issues, consumers must verify binding between VolumeSnapshot and VolumeSnapshotContent objects is successful (by validating that both VolumeSnapshot and VolumeSnapshotContent point at each other) before using this object.' + type: string + creationTime: + description: creationTime is the timestamp when the point-in-time snapshot is taken by the underlying storage system. In dynamic snapshot creation case, this field will be filled in by the snapshot controller with the "creation_time" value returned from CSI "CreateSnapshot" gRPC call. For a pre-existing snapshot, this field will be filled with the "creation_time" value returned from the CSI "ListSnapshots" gRPC call if the driver supports it. If not specified, it may indicate that the creation time of the snapshot is unknown. + format: date-time + type: string + error: + description: error is the last observed error during snapshot creation, if any. This field could be helpful to upper level controllers(i.e., application controller) to decide whether they should continue on waiting for the snapshot to be created based on the type of error reported. The snapshot controller will keep retrying when an error occurrs during the snapshot creation. Upon success, this error field will be cleared. + properties: + message: + description: 'message is a string detailing the encountered error during snapshot creation if specified. NOTE: message may be logged, and it should not contain sensitive information.' + type: string + time: + description: time is the timestamp when the error was encountered. + format: date-time + type: string + type: object + readyToUse: + description: readyToUse indicates if the snapshot is ready to be used to restore a volume. In dynamic snapshot creation case, this field will be filled in by the snapshot controller with the "ready_to_use" value returned from CSI "CreateSnapshot" gRPC call. For a pre-existing snapshot, this field will be filled with the "ready_to_use" value returned from the CSI "ListSnapshots" gRPC call if the driver supports it, otherwise, this field will be set to "True". If not specified, it means the readiness of a snapshot is unknown. + type: boolean + restoreSize: + type: string + description: restoreSize represents the minimum size of volume required to create a volume from this snapshot. In dynamic snapshot creation case, this field will be filled in by the snapshot controller with the "size_bytes" value returned from CSI "CreateSnapshot" gRPC call. For a pre-existing snapshot, this field will be filled with the "size_bytes" value returned from the CSI "ListSnapshots" gRPC call if the driver supports it. When restoring a volume from this snapshot, the size of the volume MUST NOT be smaller than the restoreSize if it is specified, otherwise the restoration will fail. If not specified, it indicates that the size is unknown. + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + required: + - spec + type: object + served: true + storage: true + subresources: + status: {} + - additionalPrinterColumns: + - description: Indicates if the snapshot is ready to be used to restore a volume. + jsonPath: .status.readyToUse + name: ReadyToUse + type: boolean + - description: If a new snapshot needs to be created, this contains the name of the source PVC from which this snapshot was (or will be) created. + jsonPath: .spec.source.persistentVolumeClaimName + name: SourcePVC + type: string + - description: If a snapshot already exists, this contains the name of the existing VolumeSnapshotContent object representing the existing snapshot. + jsonPath: .spec.source.volumeSnapshotContentName + name: SourceSnapshotContent + type: string + - description: Represents the minimum size of volume required to rehydrate from this snapshot. + jsonPath: .status.restoreSize + name: RestoreSize + type: string + - description: The name of the VolumeSnapshotClass requested by the VolumeSnapshot. + jsonPath: .spec.volumeSnapshotClassName + name: SnapshotClass + type: string + - description: Name of the VolumeSnapshotContent object to which the VolumeSnapshot object intends to bind to. Please note that verification of binding actually requires checking both VolumeSnapshot and VolumeSnapshotContent to ensure both are pointing at each other. Binding MUST be verified prior to usage of this object. + jsonPath: .status.boundVolumeSnapshotContentName + name: SnapshotContent + type: string + - description: Timestamp when the point-in-time snapshot was taken by the underlying storage system. + jsonPath: .status.creationTime + name: CreationTime + type: date + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1beta1 + # This indicates the v1beta1 version of the custom resource is deprecated. + # API requests to this version receive a warning in the server response. + deprecated: true + # This overrides the default warning returned to clients making v1beta1 API requests. + deprecationWarning: "snapshot.storage.k8s.io/v1beta1 VolumeSnapshot is deprecated; use snapshot.storage.k8s.io/v1 VolumeSnapshot" + schema: + openAPIV3Schema: + description: VolumeSnapshot is a user's request for either creating a point-in-time snapshot of a persistent volume, or binding to a pre-existing snapshot. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation of an object. Servers should convert recognized schemas to the latest internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this object represents. Servers may infer this from the endpoint the client submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + spec: + description: 'spec defines the desired characteristics of a snapshot requested by a user. More info: https://kubernetes.io/docs/concepts/storage/volume-snapshots#volumesnapshots Required.' + properties: + source: + description: source specifies where a snapshot will be created from. This field is immutable after creation. Required. + properties: + persistentVolumeClaimName: + description: persistentVolumeClaimName specifies the name of the PersistentVolumeClaim object representing the volume from which a snapshot should be created. This PVC is assumed to be in the same namespace as the VolumeSnapshot object. This field should be set if the snapshot does not exists, and needs to be created. This field is immutable. + type: string + volumeSnapshotContentName: + description: volumeSnapshotContentName specifies the name of a pre-existing VolumeSnapshotContent object representing an existing volume snapshot. This field should be set if the snapshot already exists and only needs a representation in Kubernetes. This field is immutable. + type: string + type: object + volumeSnapshotClassName: + description: 'VolumeSnapshotClassName is the name of the VolumeSnapshotClass requested by the VolumeSnapshot. VolumeSnapshotClassName may be left nil to indicate that the default SnapshotClass should be used. A given cluster may have multiple default Volume SnapshotClasses: one default per CSI Driver. If a VolumeSnapshot does not specify a SnapshotClass, VolumeSnapshotSource will be checked to figure out what the associated CSI Driver is, and the default VolumeSnapshotClass associated with that CSI Driver will be used. If more than one VolumeSnapshotClass exist for a given CSI Driver and more than one have been marked as default, CreateSnapshot will fail and generate an event. Empty string is not allowed for this field.' + type: string + required: + - source + type: object + status: + description: status represents the current information of a snapshot. Consumers must verify binding between VolumeSnapshot and VolumeSnapshotContent objects is successful (by validating that both VolumeSnapshot and VolumeSnapshotContent point at each other) before using this object. + properties: + boundVolumeSnapshotContentName: + description: 'boundVolumeSnapshotContentName is the name of the VolumeSnapshotContent object to which this VolumeSnapshot object intends to bind to. If not specified, it indicates that the VolumeSnapshot object has not been successfully bound to a VolumeSnapshotContent object yet. NOTE: To avoid possible security issues, consumers must verify binding between VolumeSnapshot and VolumeSnapshotContent objects is successful (by validating that both VolumeSnapshot and VolumeSnapshotContent point at each other) before using this object.' + type: string + creationTime: + description: creationTime is the timestamp when the point-in-time snapshot is taken by the underlying storage system. In dynamic snapshot creation case, this field will be filled in by the snapshot controller with the "creation_time" value returned from CSI "CreateSnapshot" gRPC call. For a pre-existing snapshot, this field will be filled with the "creation_time" value returned from the CSI "ListSnapshots" gRPC call if the driver supports it. If not specified, it may indicate that the creation time of the snapshot is unknown. + format: date-time + type: string + error: + description: error is the last observed error during snapshot creation, if any. This field could be helpful to upper level controllers(i.e., application controller) to decide whether they should continue on waiting for the snapshot to be created based on the type of error reported. The snapshot controller will keep retrying when an error occurrs during the snapshot creation. Upon success, this error field will be cleared. + properties: + message: + description: 'message is a string detailing the encountered error during snapshot creation if specified. NOTE: message may be logged, and it should not contain sensitive information.' + type: string + time: + description: time is the timestamp when the error was encountered. + format: date-time + type: string + type: object + readyToUse: + description: readyToUse indicates if the snapshot is ready to be used to restore a volume. In dynamic snapshot creation case, this field will be filled in by the snapshot controller with the "ready_to_use" value returned from CSI "CreateSnapshot" gRPC call. For a pre-existing snapshot, this field will be filled with the "ready_to_use" value returned from the CSI "ListSnapshots" gRPC call if the driver supports it, otherwise, this field will be set to "True". If not specified, it means the readiness of a snapshot is unknown. + type: boolean + restoreSize: + type: string + description: restoreSize represents the minimum size of volume required to create a volume from this snapshot. In dynamic snapshot creation case, this field will be filled in by the snapshot controller with the "size_bytes" value returned from CSI "CreateSnapshot" gRPC call. For a pre-existing snapshot, this field will be filled with the "size_bytes" value returned from the CSI "ListSnapshots" gRPC call if the driver supports it. When restoring a volume from this snapshot, the size of the volume MUST NOT be smaller than the restoreSize if it is specified, otherwise the restoration will fail. If not specified, it indicates that the size is unknown. + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + required: + - spec + type: object + served: true + storage: false + subresources: + status: {} +status: + acceptedNames: + kind: "" + plural: "" + conditions: [] + storedVersions: [] diff --git a/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/defaults/main.yml b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/defaults/main.yml new file mode 100644 index 0000000..1ee662e --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/defaults/main.yml @@ -0,0 +1,2 @@ +--- +gcp_pd_csi_controller_replicas: 1 diff --git a/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/tasks/main.yml b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/tasks/main.yml new file mode 100644 index 0000000..be511ca --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/tasks/main.yml @@ -0,0 +1,47 @@ +--- +- name: GCP PD CSI Driver | Check if cloud-sa.json exists + fail: + msg: "Credentials file cloud-sa.json is mandatory" + when: gcp_pd_csi_sa_cred_file is not defined or not gcp_pd_csi_sa_cred_file + +- name: GCP PD CSI Driver | Copy GCP credentials file + copy: + src: "{{ gcp_pd_csi_sa_cred_file }}" + dest: "{{ kube_config_dir }}/cloud-sa.json" + group: "{{ kube_cert_group }}" + mode: 0640 + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: GCP PD CSI Driver | Get base64 cloud-sa.json + slurp: + src: "{{ kube_config_dir }}/cloud-sa.json" + register: gcp_cred_secret + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: GCP PD CSI Driver | Generate Manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: gcp-pd-csi-cred-secret, file: gcp-pd-csi-cred-secret.yml} + - {name: gcp-pd-csi-setup, file: gcp-pd-csi-setup.yml} + - {name: gcp-pd-csi-controller, file: gcp-pd-csi-controller.yml} + - {name: gcp-pd-csi-node, file: gcp-pd-csi-node.yml} + - {name: gcp-pd-csi-sc-regional, file: gcp-pd-csi-sc-regional.yml} + - {name: gcp-pd-csi-sc-zonal, file: gcp-pd-csi-sc-zonal.yml} + register: gcp_pd_csi_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: GCP PD CSI Driver | Apply Manifests + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ gcp_pd_csi_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item.file }}" diff --git a/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-controller.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-controller.yml.j2 new file mode 100644 index 0000000..61157d8 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-controller.yml.j2 @@ -0,0 +1,165 @@ +kind: Deployment +apiVersion: apps/v1 +metadata: + name: csi-gce-pd-controller + namespace: kube-system +spec: + replicas: {{ gcp_pd_csi_controller_replicas }} + selector: + matchLabels: + app: gcp-compute-persistent-disk-csi-driver + template: + metadata: + labels: + app: gcp-compute-persistent-disk-csi-driver + spec: + # Host network must be used for interaction with Workload Identity in GKE + # since it replaces GCE Metadata Server with GKE Metadata Server. Remove + # this requirement when issue is resolved and before any exposure of + # metrics ports + hostNetwork: true + nodeSelector: + kubernetes.io/os: linux + serviceAccountName: csi-gce-pd-controller-sa + priorityClassName: csi-gce-pd-controller + containers: + - name: csi-provisioner + image: {{ csi_provisioner_image_repo }}:{{ csi_provisioner_image_tag }} + args: + - "--v=5" + - "--csi-address=/csi/csi.sock" + - "--feature-gates=Topology=true" + - "--http-endpoint=:22011" + - "--leader-election-namespace=$(PDCSI_NAMESPACE)" + - "--timeout=250s" + - "--extra-create-metadata" + # - "--run-controller-service=false" # disable the controller service of the CSI driver + # - "--run-node-service=false" # disable the node service of the CSI driver + - "--leader-election" + - "--default-fstype=ext4" + - "--controller-publish-readonly=true" + env: + - name: PDCSI_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + ports: + - containerPort: 22011 + name: http-endpoint + protocol: TCP + livenessProbe: + failureThreshold: 1 + httpGet: + path: /healthz/leader-election + port: http-endpoint + initialDelaySeconds: 10 + timeoutSeconds: 10 + periodSeconds: 20 + volumeMounts: + - name: socket-dir + mountPath: /csi + - name: csi-attacher + image: {{ csi_attacher_image_repo }}:{{ csi_attacher_image_tag }} + args: + - "--v=5" + - "--csi-address=/csi/csi.sock" + - "--http-endpoint=:22012" + - "--leader-election" + - "--leader-election-namespace=$(PDCSI_NAMESPACE)" + - "--timeout=250s" + env: + - name: PDCSI_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + ports: + - containerPort: 22012 + name: http-endpoint + protocol: TCP + livenessProbe: + failureThreshold: 1 + httpGet: + path: /healthz/leader-election + port: http-endpoint + initialDelaySeconds: 10 + timeoutSeconds: 10 + periodSeconds: 20 + volumeMounts: + - name: socket-dir + mountPath: /csi + - name: csi-resizer + image: {{ csi_resizer_image_repo }}:{{ csi_resizer_image_tag }} + args: + - "--v=5" + - "--csi-address=/csi/csi.sock" + - "--http-endpoint=:22013" + - "--leader-election" + - "--leader-election-namespace=$(PDCSI_NAMESPACE)" + - "--handle-volume-inuse-error=false" + env: + - name: PDCSI_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + ports: + - containerPort: 22013 + name: http-endpoint + protocol: TCP + livenessProbe: + failureThreshold: 1 + httpGet: + path: /healthz/leader-election + port: http-endpoint + initialDelaySeconds: 10 + timeoutSeconds: 10 + periodSeconds: 20 + volumeMounts: + - name: socket-dir + mountPath: /csi + - name: csi-snapshotter + image: {{ csi_snapshotter_image_repo }}:{{ csi_snapshotter_image_tag }} + args: + - "--v=5" + - "--csi-address=/csi/csi.sock" + - "--metrics-address=:22014" + - "--leader-election" + - "--leader-election-namespace=$(PDCSI_NAMESPACE)" + - "--timeout=300s" + env: + - name: PDCSI_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + volumeMounts: + - name: socket-dir + mountPath: /csi + - name: gce-pd-driver + # Don't change base image without changing pdImagePlaceholder in + # test/k8s-integration/main.go + image: {{ gcp_pd_csi_plugin_image_repo }}:{{ gcp_pd_csi_plugin_image_tag }} + args: + - "--v=5" + - "--endpoint=unix:/csi/csi.sock" + env: + - name: GOOGLE_APPLICATION_CREDENTIALS + value: "/etc/cloud-sa/cloud-sa.json" + volumeMounts: + - name: socket-dir + mountPath: /csi + - name: cloud-sa-volume + readOnly: true + mountPath: "/etc/cloud-sa" + volumes: + - name: socket-dir + emptyDir: {} + - name: cloud-sa-volume + secret: + secretName: cloud-sa +--- +apiVersion: storage.k8s.io/v1 +kind: CSIDriver +metadata: + name: pd.csi.storage.gke.io +spec: + attachRequired: true + podInfoOnMount: false \ No newline at end of file diff --git a/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-cred-secret.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-cred-secret.yml.j2 new file mode 100644 index 0000000..f8291a4 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-cred-secret.yml.j2 @@ -0,0 +1,8 @@ +--- +kind: Secret +apiVersion: v1 +metadata: + name: cloud-sa + namespace: kube-system +data: + cloud-sa.json: {{ gcp_cred_secret.content }} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-node.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-node.yml.j2 new file mode 100644 index 0000000..9aad620 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-node.yml.j2 @@ -0,0 +1,112 @@ +kind: DaemonSet +apiVersion: apps/v1 +metadata: + name: csi-gce-pd-node + namespace: kube-system +spec: + selector: + matchLabels: + app: gcp-compute-persistent-disk-csi-driver + template: + metadata: + labels: + app: gcp-compute-persistent-disk-csi-driver + spec: + # Host network must be used for interaction with Workload Identity in GKE + # since it replaces GCE Metadata Server with GKE Metadata Server. Remove + # this requirement when issue is resolved and before any exposure of + # metrics ports. + hostNetwork: true + priorityClassName: csi-gce-pd-node + serviceAccountName: csi-gce-pd-node-sa + containers: + - name: csi-driver-registrar + image: {{ csi_node_driver_registrar_image_repo }}:{{ csi_node_driver_registrar_image_tag }} + args: + - "--v=5" + - "--csi-address=/csi/csi.sock" + - "--kubelet-registration-path=/var/lib/kubelet/plugins/pd.csi.storage.gke.io/csi.sock" + lifecycle: + preStop: + exec: + command: ["/bin/sh", "-c", "rm -rf /registration/pd.csi.storage.gke.io /registration/pd.csi.storage.gke.io-reg.sock"] + env: + - name: KUBE_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + volumeMounts: + - name: plugin-dir + mountPath: /csi + - name: registration-dir + mountPath: /registration + - name: gce-pd-driver + securityContext: + privileged: true + # Don't change base image without changing pdImagePlaceholder in + # test/k8s-integration/main.go + image: {{ gcp_pd_csi_plugin_image_repo }}:{{ gcp_pd_csi_plugin_image_tag }} + args: + - "--v=5" + - "--endpoint=unix:/csi/csi.sock" + - "--run-controller-service=false" + volumeMounts: + - name: kubelet-dir + mountPath: /var/lib/kubelet + mountPropagation: "Bidirectional" + - name: plugin-dir + mountPath: /csi + - name: device-dir + mountPath: /dev + # The following mounts are required to trigger host udevadm from + # container + - name: udev-rules-etc + mountPath: /etc/udev + - name: udev-rules-lib + mountPath: /lib/udev + - name: udev-socket + mountPath: /run/udev + - name: sys + mountPath: /sys + nodeSelector: + kubernetes.io/os: linux + volumes: + - name: registration-dir + hostPath: + path: /var/lib/kubelet/plugins_registry/ + type: Directory + - name: kubelet-dir + hostPath: + path: /var/lib/kubelet + type: Directory + - name: plugin-dir + hostPath: + path: /var/lib/kubelet/plugins/pd.csi.storage.gke.io/ + type: DirectoryOrCreate + - name: device-dir + hostPath: + path: /dev + type: Directory + # The following mounts are required to trigger host udevadm from + # container + - name: udev-rules-etc + hostPath: + path: /etc/udev + type: Directory + - name: udev-rules-lib + hostPath: + path: /lib/udev + type: Directory + - name: udev-socket + hostPath: + path: /run/udev + type: Directory + - name: sys + hostPath: + path: /sys + type: Directory + # https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ + # See "special case". This will tolerate everything. Node component should + # be scheduled on all nodes. + tolerations: + - operator: Exists \ No newline at end of file diff --git a/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-sc-regional.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-sc-regional.yml.j2 new file mode 100644 index 0000000..57a8675 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-sc-regional.yml.j2 @@ -0,0 +1,9 @@ +apiVersion: storage.k8s.io/v1 +kind: StorageClass +metadata: + name: csi-gce-pd-regional +provisioner: pd.csi.storage.gke.io +parameters: + type: pd-balanced + replication-type: regional-pd +volumeBindingMode: WaitForFirstConsumer \ No newline at end of file diff --git a/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-sc-zonal.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-sc-zonal.yml.j2 new file mode 100644 index 0000000..e9bedaf --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-sc-zonal.yml.j2 @@ -0,0 +1,8 @@ +apiVersion: storage.k8s.io/v1 +kind: StorageClass +metadata: + name: csi-gce-pd-zonal +provisioner: pd.csi.storage.gke.io +parameters: + type: pd-balanced +volumeBindingMode: WaitForFirstConsumer \ No newline at end of file diff --git a/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-setup.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-setup.yml.j2 new file mode 100644 index 0000000..610baf3 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/gcp_pd/templates/gcp-pd-csi-setup.yml.j2 @@ -0,0 +1,291 @@ +##### Node Service Account, Roles, RoleBindings +apiVersion: v1 +kind: ServiceAccount +metadata: + name: csi-gce-pd-node-sa + namespace: kube-system + +--- +##### Controller Service Account, Roles, Rolebindings +apiVersion: v1 +kind: ServiceAccount +metadata: + name: csi-gce-pd-controller-sa + namespace: kube-system + +--- +# xref: https://github.com/kubernetes-csi/external-provisioner/blob/master/deploy/kubernetes/rbac.yaml +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-gce-pd-provisioner-role +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "create", "delete"] + - apiGroups: [""] + resources: ["persistentvolumeclaims"] + verbs: ["get", "list", "watch", "update"] + - apiGroups: ["storage.k8s.io"] + resources: ["storageclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["list", "watch", "create", "update", "patch"] + - apiGroups: ["storage.k8s.io"] + resources: ["csinodes"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["nodes"] + verbs: ["get", "list", "watch"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshots"] + verbs: ["get", "list"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotcontents"] + verbs: ["get", "list"] + # Access to volumeattachments is only needed when the CSI driver + # has the PUBLISH_UNPUBLISH_VOLUME controller capability. + # In that case, external-provisioner will watch volumeattachments + # to determine when it is safe to delete a volume. + - apiGroups: ["storage.k8s.io"] + resources: ["volumeattachments"] + verbs: ["get", "list", "watch"] +--- + +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-gce-pd-controller-provisioner-binding +subjects: + - kind: ServiceAccount + name: csi-gce-pd-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-gce-pd-provisioner-role + apiGroup: rbac.authorization.k8s.io + +--- +# xref: https://github.com/kubernetes-csi/external-attacher/blob/master/deploy/kubernetes/rbac.yaml +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-gce-pd-attacher-role +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "update", "patch"] + - apiGroups: [""] + resources: ["nodes"] + verbs: ["get", "list", "watch"] + - apiGroups: ["storage.k8s.io"] + resources: ["csinodes"] + verbs: ["get", "list", "watch"] + - apiGroups: ["storage.k8s.io"] + resources: ["volumeattachments"] + verbs: ["get", "list", "watch", "update", "patch"] + - apiGroups: ["storage.k8s.io"] + resources: ["volumeattachments/status"] + verbs: ["patch"] +--- + +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-gce-pd-controller-attacher-binding +subjects: + - kind: ServiceAccount + name: csi-gce-pd-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-gce-pd-attacher-role + apiGroup: rbac.authorization.k8s.io + +--- + +apiVersion: scheduling.k8s.io/v1 +kind: PriorityClass +metadata: + name: csi-gce-pd-controller +value: 900000000 +globalDefault: false +description: "This priority class should be used for the GCE PD CSI driver controller deployment only." + +--- + +apiVersion: scheduling.k8s.io/v1 +kind: PriorityClass +metadata: + name: csi-gce-pd-node +value: 900001000 +globalDefault: false +description: "This priority class should be used for the GCE PD CSI driver node deployment only." + +--- + +# Resizer must be able to work with PVCs, PVs, SCs. +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-gce-pd-resizer-role +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "update", "patch"] + - apiGroups: [""] + resources: ["persistentvolumeclaims"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["persistentvolumeclaims/status"] + verbs: ["update", "patch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["list", "watch", "create", "update", "patch"] + # If handle-volume-inuse-error=true, the pod specific rbac is needed + - apiGroups: [""] + resources: ["pods"] + verbs: ["get", "list", "watch"] + +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-gce-pd-resizer-binding +subjects: + - kind: ServiceAccount + name: csi-gce-pd-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-gce-pd-resizer-role + apiGroup: rbac.authorization.k8s.io +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-gce-pd-controller-deploy +rules: + - apiGroups: ["policy"] + resources: ["podsecuritypolicies"] + verbs: ["use"] + resourceNames: + - csi-gce-pd-controller-psp +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: csi-gce-pd-controller-deploy +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: csi-gce-pd-controller-deploy +subjects: + - kind: ServiceAccount + name: csi-gce-pd-controller-sa + namespace: kube-system +--- + +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-gce-pd-node-deploy +rules: + - apiGroups: ['policy'] + resources: ['podsecuritypolicies'] + verbs: ['use'] + resourceNames: + - csi-gce-pd-node-psp +--- + +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: csi-gce-pd-node +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: csi-gce-pd-node-deploy +subjects: +- kind: ServiceAccount + name: csi-gce-pd-node-sa + namespace: kube-system +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: csi-gce-pd-controller +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: csi-gce-pd-node-deploy +subjects: +- kind: ServiceAccount + name: csi-gce-pd-controller-sa + namespace: kube-system + +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: csi-gce-pd-snapshotter-role +rules: + - apiGroups: [""] + resources: ["events"] + verbs: ["list", "watch", "create", "update", "patch"] + # Secrets resource omitted since GCE PD snapshots does not require them + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotcontents"] + verbs: ["create", "get", "list", "watch", "update", "delete", "patch"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotcontents/status"] + verbs: ["update", "patch"] +--- + +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-gce-pd-controller-snapshotter-binding +subjects: + - kind: ServiceAccount + name: csi-gce-pd-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-gce-pd-snapshotter-role + apiGroup: rbac.authorization.k8s.io +--- + +kind: Role +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-gce-pd-leaderelection-role + namespace: kube-system + labels: + k8s-app: gcp-compute-persistent-disk-csi-driver +rules: +- apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "watch", "list", "delete", "update", "create"] + +--- + +kind: RoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-gce-pd-controller-leaderelection-binding + namespace: kube-system + labels: + k8s-app: gcp-compute-persistent-disk-csi-driver +subjects: +- kind: ServiceAccount + name: csi-gce-pd-controller-sa + namespace: kube-system +roleRef: + kind: Role + name: csi-gce-pd-leaderelection-role + apiGroup: rbac.authorization.k8s.io \ No newline at end of file diff --git a/kubespray/roles/kubernetes-apps/csi_driver/upcloud/defaults/main.yml b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/defaults/main.yml new file mode 100644 index 0000000..ea828f3 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/defaults/main.yml @@ -0,0 +1,16 @@ +--- +upcloud_csi_controller_replicas: 1 +upcloud_csi_provisioner_image_tag: "v3.1.0" +upcloud_csi_attacher_image_tag: "v3.4.0" +upcloud_csi_resizer_image_tag: "v1.4.0" +upcloud_csi_plugin_image_tag: "v0.3.3" +upcloud_csi_node_image_tag: "v2.5.0" +upcloud_username: "{{ lookup('env', 'UPCLOUD_USERNAME') }}" +upcloud_password: "{{ lookup('env', 'UPCLOUD_PASSWORD') }}" +upcloud_tolerations: [] +upcloud_csi_enable_volume_snapshot: false +upcloud_csi_snapshot_controller_replicas: 2 +upcloud_csi_snapshotter_image_tag: "v4.2.1" +upcloud_csi_snapshot_controller_image_tag: "v4.2.1" +upcloud_csi_snapshot_validation_webhook_image_tag: "v4.2.1" +upcloud_cacert: "{{ lookup('env', 'OS_CACERT') }}" diff --git a/kubespray/roles/kubernetes-apps/csi_driver/upcloud/tasks/main.yml b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/tasks/main.yml new file mode 100644 index 0000000..8f0b69f --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/tasks/main.yml @@ -0,0 +1,40 @@ +--- +- name: UpCloud CSI Driver | Check if UPCLOUD_USERNAME exists + fail: + msg: "UpCloud username is missing. Env UPCLOUD_USERNAME is mandatory" + when: upcloud_username is not defined or not upcloud_username + +- name: UpCloud CSI Driver | Check if UPCLOUD_PASSWORD exists + fail: + msg: "UpCloud password is missing. Env UPCLOUD_PASSWORD is mandatory" + when: + - upcloud_username is defined + - upcloud_username | length > 0 + - upcloud_password is not defined or not upcloud_password + +- name: UpCloud CSI Driver | Generate Manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: upcloud-csi-cred-secret, file: upcloud-csi-cred-secret.yml} + - {name: upcloud-csi-setup, file: upcloud-csi-setup.yml} + - {name: upcloud-csi-controller, file: upcloud-csi-controller.yml} + - {name: upcloud-csi-node, file: upcloud-csi-node.yml} + - {name: upcloud-csi-driver, file: upcloud-csi-driver.yml} + register: upcloud_csi_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: UpCloud CSI Driver | Apply Manifests + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ upcloud_csi_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item.file }}" diff --git a/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-controller.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-controller.yml.j2 new file mode 100644 index 0000000..1b8519d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-controller.yml.j2 @@ -0,0 +1,93 @@ +kind: StatefulSet +apiVersion: apps/v1 +metadata: + name: csi-upcloud-controller + namespace: kube-system +spec: + serviceName: "csi-upcloud" + replicas: {{ upcloud_csi_controller_replicas }} + selector: + matchLabels: + app: csi-upcloud-controller + template: + metadata: + labels: + app: csi-upcloud-controller + role: csi-upcloud + spec: + priorityClassName: system-cluster-critical + serviceAccount: csi-upcloud-controller-sa + containers: + - name: csi-provisioner + image: registry.k8s.io/sig-storage/csi-provisioner:{{ upcloud_csi_provisioner_image_tag }} + args: + - "--csi-address=$(ADDRESS)" + - "--v=5" + - "--timeout=600s" + env: + - name: ADDRESS + value: /var/lib/csi/sockets/pluginproxy/csi.sock + imagePullPolicy: "Always" + volumeMounts: + - name: socket-dir + mountPath: /var/lib/csi/sockets/pluginproxy/ + - name: csi-attacher + image: registry.k8s.io/sig-storage/csi-attacher:{{ upcloud_csi_attacher_image_tag }} + args: + - "--v=5" + - "--csi-address=$(ADDRESS)" + - "--timeout=120s" + env: + - name: ADDRESS + value: /var/lib/csi/sockets/pluginproxy/csi.sock + imagePullPolicy: "Always" + volumeMounts: + - name: socket-dir + mountPath: /var/lib/csi/sockets/pluginproxy/ + - name: csi-resizer + image: registry.k8s.io/sig-storage/csi-resizer:{{ upcloud_csi_resizer_image_tag }} + args: + - "--v=5" + - "--timeout=120s" + - "--csi-address=$(ADDRESS)" + - "--handle-volume-inuse-error=true" + env: + - name: ADDRESS + value: /var/lib/csi/sockets/pluginproxy/csi.sock + imagePullPolicy: "Always" + volumeMounts: + - name: socket-dir + mountPath: /var/lib/csi/sockets/pluginproxy/ + - name: csi-upcloud-plugin + image: ghcr.io/upcloudltd/upcloud-csi:{{ upcloud_csi_plugin_image_tag }} + args: + - "--endpoint=$(CSI_ENDPOINT)" + - "--nodehost=$(NODE_ID)" + - "--username=$(UPCLOUD_USERNAME)" + - "--password=$(UPCLOUD_PASSWORD)" + env: + - name: CSI_ENDPOINT + value: unix:///var/lib/csi/sockets/pluginproxy/csi.sock + - name: UPCLOUD_USERNAME + valueFrom: + secretKeyRef: + name: upcloud + key: username + - name: UPCLOUD_PASSWORD + valueFrom: + secretKeyRef: + name: upcloud + key: password + - name: NODE_ID + valueFrom: + fieldRef: + fieldPath: spec.nodeName + imagePullPolicy: "Always" + volumeMounts: + - name: socket-dir + mountPath: /var/lib/csi/sockets/pluginproxy/ + imagePullSecrets: + - name: regcred + volumes: + - name: socket-dir + emptyDir: {} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-cred-secret.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-cred-secret.yml.j2 new file mode 100644 index 0000000..5e91d88 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-cred-secret.yml.j2 @@ -0,0 +1,9 @@ +--- +apiVersion: v1 +kind: Secret +metadata: + name: upcloud + namespace: kube-system +stringData: + username: {{ upcloud_username }} + password: {{ upcloud_password }} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-driver.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-driver.yml.j2 new file mode 100644 index 0000000..8f4c612 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-driver.yml.j2 @@ -0,0 +1,8 @@ +apiVersion: storage.k8s.io/v1 +kind: CSIDriver +metadata: + name: storage.csi.upcloud.com +spec: + attachRequired: true + podInfoOnMount: true + fsGroupPolicy: File diff --git a/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-node.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-node.yml.j2 new file mode 100644 index 0000000..7ed39be --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-node.yml.j2 @@ -0,0 +1,101 @@ +kind: DaemonSet +apiVersion: apps/v1 +metadata: + name: csi-upcloud-node + namespace: kube-system +spec: + selector: + matchLabels: + app: csi-upcloud-node + template: + metadata: + labels: + app: csi-upcloud-node + role: csi-upcloud + spec: + priorityClassName: system-node-critical + serviceAccount: csi-upcloud-node-sa + hostNetwork: true + containers: + - name: csi-node-driver-registrar + image: registry.k8s.io/sig-storage/csi-node-driver-registrar:{{ upcloud_csi_node_image_tag }} + args: + - "--v=5" + - "--csi-address=$(ADDRESS)" + - "--kubelet-registration-path=$(DRIVER_REG_SOCK_PATH)" + env: + - name: ADDRESS + value: /csi/csi.sock + - name: DRIVER_REG_SOCK_PATH + value: /var/lib/kubelet/plugins/storage.csi.upcloud.com/csi.sock + - name: KUBE_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + volumeMounts: + - name: plugin-dir + mountPath: /csi/ + - name: registration-dir + mountPath: /registration/ + - name: csi-upcloud-plugin + image: ghcr.io/upcloudltd/upcloud-csi:{{ upcloud_csi_plugin_image_tag }} + args: + - "--endpoint=$(CSI_ENDPOINT)" + - "--nodehost=$(NODE_ID)" + - "--username=$(UPCLOUD_USERNAME)" + - "--password=$(UPCLOUD_PASSWORD)" + env: + - name: CSI_ENDPOINT + value: unix:///csi/csi.sock + - name: UPCLOUD_USERNAME + valueFrom: + secretKeyRef: + name: upcloud + key: username + - name: UPCLOUD_PASSWORD + valueFrom: + secretKeyRef: + name: upcloud + key: password + - name: NODE_ID + valueFrom: + fieldRef: + fieldPath: spec.nodeName + imagePullPolicy: "Always" + securityContext: + privileged: true + capabilities: + add: ["SYS_ADMIN"] + allowPrivilegeEscalation: true + volumeMounts: + - name: plugin-dir + mountPath: /csi + - name: pods-mount-dir + mountPath: /var/lib/kubelet + # needed so that any mounts setup inside this container are + # propagated back to the host machine. + mountPropagation: "Bidirectional" + - name: device-dir + mountPath: /dev + imagePullSecrets: + - name: regcred + volumes: + - name: registration-dir + hostPath: + path: /var/lib/kubelet/plugins_registry/ + type: DirectoryOrCreate + - name: plugin-dir + hostPath: + path: /var/lib/kubelet/plugins/storage.csi.upcloud.com + type: DirectoryOrCreate + - name: pods-mount-dir + hostPath: + path: /var/lib/kubelet + type: Directory + - name: device-dir + hostPath: + path: /dev +{% if upcloud_tolerations %} + tolerations: + {{ upcloud_tolerations | to_nice_yaml(indent=2) | indent(width=8) }} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-setup.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-setup.yml.j2 new file mode 100644 index 0000000..5af71d2 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/upcloud/templates/upcloud-csi-setup.yml.j2 @@ -0,0 +1,185 @@ +kind: ServiceAccount +apiVersion: v1 +metadata: + name: csi-upcloud-controller-sa + namespace: kube-system + +--- + +apiVersion: v1 +kind: ServiceAccount +metadata: + name: csi-upcloud-node-sa + namespace: kube-system + +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-upcloud-node-driver-registrar-role + namespace: kube-system +rules: + - apiGroups: [ "" ] + resources: [ "events" ] + verbs: [ "get", "list", "watch", "create", "update", "patch" ] + +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-upcloud-node-driver-registrar-binding +subjects: + - kind: ServiceAccount + name: csi-upcloud-node-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-upcloud-node-driver-registrar-role + apiGroup: rbac.authorization.k8s.io + +--- + +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-upcloud-provisioner-role +rules: + - apiGroups: [ "" ] + resources: [ "secrets" ] + verbs: [ "get", "list" ] + - apiGroups: [ "" ] + resources: [ "persistentvolumes" ] + verbs: [ "get", "list", "watch", "create", "delete" ] + - apiGroups: [ "" ] + resources: [ "persistentvolumeclaims" ] + verbs: [ "get", "list", "watch", "update" ] + - apiGroups: [ "storage.k8s.io" ] + resources: [ "storageclasses" ] + verbs: [ "get", "list", "watch" ] + - apiGroups: [ "storage.k8s.io" ] + resources: [ "csinodes" ] + verbs: [ "get", "list", "watch" ] + - apiGroups: [ "" ] + resources: [ "events" ] + verbs: [ "list", "watch", "create", "update", "patch" ] + - apiGroups: [ "" ] + resources: [ "nodes" ] + verbs: [ "get", "list", "watch" ] + +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-upcloud-provisioner-binding +subjects: + - kind: ServiceAccount + name: csi-upcloud-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-upcloud-provisioner-role + apiGroup: rbac.authorization.k8s.io + +--- +# Attacher must be able to work with PVs, nodes and VolumeAttachments +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-upcloud-attacher-role +rules: + - apiGroups: [ "" ] + resources: [ "persistentvolumes" ] + verbs: [ "get", "list", "watch", "update", "patch" ] + - apiGroups: [ "" ] + resources: [ "nodes" ] + verbs: [ "get", "list", "watch" ] + - apiGroups: [ "storage.k8s.io" ] + resources: [ "csinodes" ] + verbs: [ "get", "list", "watch" ] + - apiGroups: [ "storage.k8s.io" ] + resources: [ "volumeattachments" ] + verbs: [ "get", "list", "watch", "update", "patch" ] + - apiGroups: [ "storage.k8s.io" ] + resources: [ "volumeattachments/status" ] + verbs: [ "get", "list", "watch", "update", "patch" ] + +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-upcloud-attacher-binding +subjects: + - kind: ServiceAccount + name: csi-upcloud-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-upcloud-attacher-role + apiGroup: rbac.authorization.k8s.io + +--- +# Provisioner must be able to work with endpoints and leases in current namespace +# if (and only if) leadership election is enabled +kind: Role +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + namespace: kube-system + name: csi-upcloud-provisioner-cfg-role +rules: +- apiGroups: [""] + resources: ["endpoints"] + verbs: ["get", "watch", "list", "delete", "update", "create"] +- apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "watch", "list", "delete", "update", "create"] + +--- +kind: RoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-provisioner-role-cfg-binding + namespace: kube-system +subjects: + - kind: ServiceAccount + name: csi-upcloud-controller-sa + namespace: kube-system +roleRef: + kind: Role + name: csi-upcloud-provisioner-cfg-role + apiGroup: rbac.authorization.k8s.io + +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-upcloud-resizer-role +rules: + - apiGroups: [ "" ] + resources: [ "persistentvolumes" ] + verbs: [ "get", "list", "watch", "update", "patch" ] + - apiGroups: [ "" ] + resources: [ "persistentvolumeclaims" ] + verbs: [ "get", "list", "watch" ] + - apiGroups: [ "" ] + resources: [ "persistentvolumeclaims/status" ] + verbs: [ "update", "patch" ] + - apiGroups: [ "" ] + resources: [ "events" ] + verbs: [ "list", "watch", "create", "update", "patch" ] + - apiGroups: [ "" ] + resources: [ "pods" ] + verbs: [ "watch", "list" ] + +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: csi-upcloud-resizer-binding +subjects: + - kind: ServiceAccount + name: csi-upcloud-controller-sa + namespace: kube-system +roleRef: + kind: ClusterRole + name: csi-upcloud-resizer-role + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/csi_driver/vsphere/defaults/main.yml b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/defaults/main.yml new file mode 100644 index 0000000..0d41441 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/defaults/main.yml @@ -0,0 +1,54 @@ +--- +external_vsphere_vcenter_port: "443" +external_vsphere_insecure: "true" +external_vsphere_kubernetes_cluster_id: "kubernetes-cluster-id" +external_vsphere_version: "7.0u1" + +vsphere_syncer_image_tag: "v3.1.0" +vsphere_csi_attacher_image_tag: "v4.3.0" +vsphere_csi_controller: "v3.1.0" +vsphere_csi_liveness_probe_image_tag: "v2.10.0" +vsphere_csi_provisioner_image_tag: "v3.5.0" +vsphere_csi_snapshotter_image_tag: "v6.2.2" +vsphere_csi_node_driver_registrar_image_tag: "v2.8.0" +vsphere_csi_driver_image_tag: "v3.1.0" +vsphere_csi_resizer_tag: "v1.8.0" + +# Set to kube-system for backward compatibility, should be change to vmware-system-csi on the long run +vsphere_csi_namespace: "kube-system" + +vsphere_csi_controller_replicas: 1 + +csi_endpoint: '{% if external_vsphere_version >= "7.0u1" %}/csi{% else %}/var/lib/csi/sockets/pluginproxy{% endif %}' + +vsphere_csi_aggressive_node_drain: False +vsphere_csi_aggressive_node_unreachable_timeout: 300 +vsphere_csi_aggressive_node_not_ready_timeout: 300 + +vsphere_csi_node_affinity: {} + +# If this is true, debug information will be displayed but +# may contain some private data, so it is recommended to set it to false +# in the production environment. +unsafe_show_logs: false + +# https://github.com/kubernetes-sigs/vsphere-csi-driver/blob/master/docs/book/features/volume_snapshot.md#how-to-enable-volume-snapshot--restore-feature-in-vsphere-csi- +# according to the above link , we can controler the block-volume-snapshot parameter +vsphere_csi_block_volume_snapshot: false + +external_vsphere_user: "{{ lookup('env', 'VSPHERE_USER') }}" +external_vsphere_password: "{{ lookup('env', 'VSPHERE_PASSWORD') }}" + +# Controller resources +vsphere_csi_snapshotter_resources: {} +vsphere_csi_provisioner_resources: {} +vsphere_syncer_resources: {} +vsphere_csi_liveness_probe_controller_resources: {} +vsphere_csi_resources: {} +vsphere_csi_resizer_resources: {} +vsphere_csi_attacher_resources: {} + +# DaemonSet node resources +vsphere_csi_node_driver_registrar_resources: {} +vsphere_csi_driver_resources: {} +vsphere_csi_liveness_probe_ds_resources: {} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/vsphere/tasks/main.yml b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/tasks/main.yml new file mode 100644 index 0000000..102dd8b --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/tasks/main.yml @@ -0,0 +1,55 @@ +--- +- name: VSphere CSI Driver | Check vsphare credentials + include_tasks: vsphere-credentials-check.yml + +- name: VSphere CSI Driver | Generate CSI cloud-config + template: + src: "{{ item }}.j2" + dest: "{{ kube_config_dir }}/{{ item }}" + mode: 0640 + with_items: + - vsphere-csi-cloud-config + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: VSphere CSI Driver | Generate Manifests + template: + src: "{{ item }}.j2" + dest: "{{ kube_config_dir }}/{{ item }}" + mode: 0644 + with_items: + - vsphere-csi-namespace.yml + - vsphere-csi-driver.yml + - vsphere-csi-controller-rbac.yml + - vsphere-csi-node-rbac.yml + - vsphere-csi-controller-config.yml + - vsphere-csi-controller-deployment.yml + - vsphere-csi-controller-service.yml + - vsphere-csi-node.yml + register: vsphere_csi_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: VSphere CSI Driver | Apply Manifests + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.item }}" + state: "latest" + with_items: + - "{{ vsphere_csi_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item }}" + +- name: VSphere CSI Driver | Generate a CSI secret manifest + command: "{{ kubectl }} create secret generic vsphere-config-secret --from-file=csi-vsphere.conf={{ kube_config_dir }}/vsphere-csi-cloud-config -n {{ vsphere_csi_namespace }} --dry-run --save-config -o yaml" + register: vsphere_csi_secret_manifest + when: inventory_hostname == groups['kube_control_plane'][0] + no_log: "{{ not (unsafe_show_logs | bool) }}" + +- name: VSphere CSI Driver | Apply a CSI secret manifest + command: + cmd: "{{ kubectl }} apply -f -" + stdin: "{{ vsphere_csi_secret_manifest.stdout }}" + when: inventory_hostname == groups['kube_control_plane'][0] + no_log: "{{ not (unsafe_show_logs | bool) }}" diff --git a/kubespray/roles/kubernetes-apps/csi_driver/vsphere/tasks/vsphere-credentials-check.yml b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/tasks/vsphere-credentials-check.yml new file mode 100644 index 0000000..3504f60 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/tasks/vsphere-credentials-check.yml @@ -0,0 +1,38 @@ +--- +- name: External vSphere Cloud Provider | check external_vsphere_vcenter_ip value + fail: + msg: "external_vsphere_vcenter_ip is missing" + when: external_vsphere_vcenter_ip is not defined or not external_vsphere_vcenter_ip + +- name: External vSphere Cloud Provider | check external_vsphere_vcenter_port value + fail: + msg: "external_vsphere_vcenter_port is missing" + when: external_vsphere_vcenter_port is not defined or not external_vsphere_vcenter_port + +- name: External vSphere Cloud Provider | check external_vsphere_insecure value + fail: + msg: "external_vsphere_insecure is missing" + when: external_vsphere_insecure is not defined or not external_vsphere_insecure + +- name: External vSphere Cloud Provider | check external_vsphere_user value + fail: + msg: "external_vsphere_user is missing" + when: external_vsphere_user is not defined or not external_vsphere_user + +- name: External vSphere Cloud Provider | check external_vsphere_password value + fail: + msg: "external_vsphere_password is missing" + when: + - external_vsphere_password is not defined or not external_vsphere_password + +- name: External vSphere Cloud Provider | check external_vsphere_datacenter value + fail: + msg: "external_vsphere_datacenter is missing" + when: + - external_vsphere_datacenter is not defined or not external_vsphere_datacenter + +- name: External vSphere Cloud Provider | check external_vsphere_kubernetes_cluster_id value + fail: + msg: "external_vsphere_kubernetes_cluster_id is missing" + when: + - external_vsphere_kubernetes_cluster_id is not defined or not external_vsphere_kubernetes_cluster_id diff --git a/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-cloud-config.j2 b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-cloud-config.j2 new file mode 100644 index 0000000..ee5033a --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-cloud-config.j2 @@ -0,0 +1,9 @@ +[Global] +cluster-id = "{{ external_vsphere_kubernetes_cluster_id }}" + +[VirtualCenter "{{ external_vsphere_vcenter_ip }}"] +insecure-flag = "{{ external_vsphere_insecure }}" +user = "{{ external_vsphere_user }}" +password = "{{ external_vsphere_password }}" +port = "{{ external_vsphere_vcenter_port }}" +datacenters = "{{ external_vsphere_datacenter }}" diff --git a/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-config.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-config.yml.j2 new file mode 100644 index 0000000..fb52d10 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-config.yml.j2 @@ -0,0 +1,31 @@ +apiVersion: v1 +data: +{% if external_vsphere_version >= "7.0" %} + "csi-auth-check": "true" +{% else %} + "csi-auth-check": "false" +{% endif %} + "csi-auth-check": "true" + "online-volume-extend": "true" + "trigger-csi-fullsync": "false" + "async-query-volume": "true" + "block-volume-snapshot": "true" + "csi-windows-support": "false" + "list-volumes": "true" + "pv-to-backingdiskobjectid-mapping": "false" + "cnsmgr-suspend-create-volume": "true" + "topology-preferential-datastores": "true" + "max-pvscsi-targets-per-vm": "true" + "multi-vcenter-csi-topology": "true" + "csi-internal-generated-cluster-id": "true" + "listview-tasks": "true" +{% if vsphere_csi_controller is version('v2.7.0', '>=') %} + "improved-csi-idempotency": "true" + "improved-volume-topology": "true" + "use-csinode-id": "true" + "list-volumes": "false" +{% endif %} +kind: ConfigMap +metadata: + name: internal-feature-states.csi.vsphere.vmware.com + namespace: "{{ vsphere_csi_namespace }}" diff --git a/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-deployment.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-deployment.yml.j2 new file mode 100644 index 0000000..dd009d8 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-deployment.yml.j2 @@ -0,0 +1,257 @@ +kind: Deployment +apiVersion: apps/v1 +metadata: + name: vsphere-csi-controller + namespace: "{{ vsphere_csi_namespace }}" +spec: + replicas: {{ vsphere_csi_controller_replicas }} + strategy: + type: RollingUpdate + rollingUpdate: + maxUnavailable: 1 + maxSurge: 0 + selector: + matchLabels: + app: vsphere-csi-controller + template: + metadata: + labels: + app: vsphere-csi-controller + role: vsphere-csi + spec: + priorityClassName: system-cluster-critical # Guarantees scheduling for critical system pods + affinity: + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: "app" + operator: In + values: + - vsphere-csi-controller + topologyKey: "kubernetes.io/hostname" + serviceAccountName: vsphere-csi-controller + nodeSelector: + node-role.kubernetes.io/control-plane: "" + tolerations: + - operator: "Exists" + key: node-role.kubernetes.io/master + effect: NoSchedule + - operator: "Exists" + key: node-role.kubernetes.io/control-plane + effect: NoSchedule +{% if vsphere_csi_aggressive_node_drain %} + # set below toleration if you need an aggressive pod eviction in case when + # node becomes not-ready or unreachable. Default is 300 seconds if not specified. + - key: node.kubernetes.io/not-ready + operator: Exists + effect: NoExecute + tolerationSeconds: {{ vsphere_csi_aggressive_node_not_ready_timeout }} + - key: node.kubernetes.io/unreachable + operator: Exists + effect: NoExecute + tolerationSeconds: {{ vsphere_csi_aggressive_node_unreachable_timeout }} +{% endif %} + dnsPolicy: "Default" + containers: + - name: csi-attacher + image: {{ kube_image_repo }}/sig-storage/csi-attacher:{{ vsphere_csi_attacher_image_tag }} + args: + - "--v=4" + - "--timeout=300s" + - "--csi-address=$(ADDRESS)" + - "--leader-election" + - "--leader-election-lease-duration=120s" + - "--leader-election-renew-deadline=60s" + - "--leader-election-retry-period=30s" + - "--kube-api-qps=100" + - "--kube-api-burst=100" +{% if vsphere_csi_attacher_resources | length > 0 %} + resources: + {{ vsphere_csi_attacher_resources | default({}) | to_nice_yaml | trim | indent(width=12) }} +{% endif %} + env: + - name: ADDRESS + value: /csi/csi.sock + volumeMounts: + - mountPath: /csi + name: socket-dir +{% if external_vsphere_version >= "7.0" %} + - name: csi-resizer + image: {{ kube_image_repo }}/sig-storage/csi-resizer:{{ vsphere_csi_resizer_tag }} + args: + - "--v=4" + - "--timeout=300s" + - "--csi-address=$(ADDRESS)" + - "--handle-volume-inuse-error=false" + - "--kube-api-qps=100" + - "--kube-api-burst=100" + - "--leader-election" + - "--leader-election-lease-duration=120s" + - "--leader-election-renew-deadline=60s" + - "--leader-election-retry-period=30s" +{% if vsphere_csi_resizer_resources | length > 0 %} + resources: + {{ vsphere_csi_resizer_resources | default({}) | to_nice_yaml | trim | indent(width=12) }} +{% endif %} + env: + - name: ADDRESS + value: /csi/csi.sock + volumeMounts: + - mountPath: /csi + name: socket-dir +{% endif %} + - name: vsphere-csi-controller + image: {{ gcr_image_repo }}/cloud-provider-vsphere/csi/release/driver:{{ vsphere_csi_controller }} + args: + - "--fss-name=internal-feature-states.csi.vsphere.vmware.com" + - "--fss-namespace={{ vsphere_csi_namespace }}" +{% if vsphere_csi_resources | length > 0 %} + resources: + {{ vsphere_csi_resources | default({}) | to_nice_yaml | trim | indent(width=12) }} +{% endif %} + imagePullPolicy: {{ k8s_image_pull_policy }} + env: + - name: CSI_ENDPOINT + value: unix://{{ csi_endpoint }}/csi.sock + - name: X_CSI_MODE + value: "controller" + - name: X_CSI_SPEC_DISABLE_LEN_CHECK + value: "true" + - name: X_CSI_SERIAL_VOL_ACCESS_TIMEOUT + value: 3m + - name: VSPHERE_CSI_CONFIG + value: "/etc/cloud/csi-vsphere.conf" + - name: LOGGER_LEVEL + value: "PRODUCTION" # Options: DEVELOPMENT, PRODUCTION +{% if external_vsphere_version >= "7.0u1" %} + - name: INCLUSTER_CLIENT_QPS + value: "100" + - name: INCLUSTER_CLIENT_BURST + value: "100" +{% endif %} + volumeMounts: + - mountPath: /etc/cloud + name: vsphere-config-volume + readOnly: true + - mountPath: {{ csi_endpoint }} + name: socket-dir + securityContext: + runAsNonRoot: true + runAsUser: 65532 + runAsGroup: 65532 + ports: + - name: healthz + containerPort: 9808 + protocol: TCP + - name: prometheus + containerPort: 2112 + protocol: TCP + livenessProbe: + httpGet: + path: /healthz + port: healthz + initialDelaySeconds: 30 + timeoutSeconds: 10 + periodSeconds: 180 + failureThreshold: 3 + - name: liveness-probe + image: {{ kube_image_repo }}/sig-storage/livenessprobe:{{ vsphere_csi_liveness_probe_image_tag }} + args: + - "--v=4" + - "--csi-address=$(ADDRESS)" +{% if vsphere_csi_liveness_probe_controller_resources | length > 0 %} + resources: + {{ vsphere_csi_liveness_probe_controller_resources | default({}) | to_nice_yaml | trim | indent(width=12) }} +{% endif %} + env: + - name: ADDRESS + value: {{ csi_endpoint }}/csi.sock + volumeMounts: + - name: socket-dir + mountPath: {{ csi_endpoint }} + - name: vsphere-syncer + image: {{ gcr_image_repo }}/cloud-provider-vsphere/csi/release/syncer:{{ vsphere_syncer_image_tag }} + args: + - "--leader-election" + - "--leader-election-lease-duration=30s" + - "--leader-election-renew-deadline=20s" + - "--leader-election-retry-period=10s" + - "--fss-name=internal-feature-states.csi.vsphere.vmware.com" + - "--fss-namespace={{ vsphere_csi_namespace }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + securityContext: + runAsNonRoot: true + runAsUser: 65532 + runAsGroup: 65532 + ports: + - containerPort: 2113 + name: prometheus + protocol: TCP +{% if vsphere_syncer_resources | length > 0 %} + resources: + {{ vsphere_syncer_resources | default({}) | to_nice_yaml | trim | indent(width=12) }} +{% endif %} + env: + - name: FULL_SYNC_INTERVAL_MINUTES + value: "30" + - name: VSPHERE_CSI_CONFIG + value: "/etc/cloud/csi-vsphere.conf" + - name: LOGGER_LEVEL + value: "PRODUCTION" # Options: DEVELOPMENT, PRODUCTION +{% if external_vsphere_version >= "7.0u1" %} + - name: INCLUSTER_CLIENT_QPS + value: "100" + - name: INCLUSTER_CLIENT_BURST + value: "100" +{% endif %} + volumeMounts: + - mountPath: /etc/cloud + name: vsphere-config-volume + readOnly: true + - name: csi-provisioner + image: {{ kube_image_repo }}/sig-storage/csi-provisioner:{{ vsphere_csi_provisioner_image_tag }} + args: + - "--v=4" + - "--timeout=300s" + - "--csi-address=$(ADDRESS)" + - "--kube-api-qps=100" + - "--kube-api-burst=100" + - "--leader-election" + - "--leader-election-lease-duration=120s" + - "--leader-election-renew-deadline=60s" + - "--leader-election-retry-period=30s" + - "--default-fstype=ext4" + - "--leader-election" + - "--default-fstype=ext4" + # needed only for topology aware setup + #- "--feature-gates=Topology=true" + #- "--strict-topology" +{% if vsphere_csi_provisioner_resources | length > 0 %} + resources: + {{ vsphere_csi_provisioner_resources | default({}) | to_nice_yaml | trim | indent(width=12) }} +{% endif %} + - name: csi-snapshotter + image: {{ kube_image_repo }}/sig-storage/csi-snapshotter:{{ vsphere_csi_snapshotter_image_tag }} + args: + - "--v=4" + - "--kube-api-qps=100" + - "--kube-api-burst=100" + - "--timeout=300s" + - "--csi-address=$(ADDRESS)" + - "--leader-election" + - "--leader-election-lease-duration=120s" + - "--leader-election-renew-deadline=60s" + - "--leader-election-retry-period=30s" + env: + - name: ADDRESS + value: /csi/csi.sock + volumeMounts: + - mountPath: /csi + name: socket-dir + volumes: + - name: vsphere-config-volume + secret: + secretName: vsphere-config-secret + - name: socket-dir + emptyDir: {} diff --git a/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-rbac.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-rbac.yml.j2 new file mode 100644 index 0000000..013d3dc --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-rbac.yml.j2 @@ -0,0 +1,89 @@ +kind: ServiceAccount +apiVersion: v1 +metadata: + name: vsphere-csi-controller + namespace: "{{ vsphere_csi_namespace }}" +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: vsphere-csi-controller-role +rules: + - apiGroups: [""] + resources: ["nodes", "pods"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["configmaps"] + verbs: ["get", "list", "watch", "create"] + - apiGroups: [""] + resources: ["persistentvolumeclaims"] + verbs: ["get", "list", "watch", "update"] +{% if external_vsphere_version >= "7.0" %} + - apiGroups: [""] + resources: ["persistentvolumeclaims/status"] +{% if external_vsphere_version >= "7.0u1" %} + verbs: ["patch"] +{% else %} + verbs: ["update", "patch"] +{% endif %} +{% endif %} + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "create", "update", "delete", "patch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["get", "list", "watch", "create", "update", "patch"] +{% if vsphere_csi_controller is version('v2.0.0', '>=') %} + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "watch", "list", "delete", "update", "create"] +{% endif %} + - apiGroups: ["storage.k8s.io"] + resources: ["storageclasses","csinodes"] + verbs: ["get", "list", "watch"] + - apiGroups: ["storage.k8s.io"] + resources: ["volumeattachments"] + verbs: ["get", "list", "watch", "patch", "update"] + - apiGroups: ["cns.vmware.com"] + resources: ["triggercsifullsyncs"] + verbs: ["create", "get", "update", "watch", "list"] + - apiGroups: ["cns.vmware.com"] + resources: ["cnsvspherevolumemigrations"] + verbs: ["create", "get", "list", "watch", "update", "delete"] + - apiGroups: ["apiextensions.k8s.io"] + resources: ["customresourcedefinitions"] + verbs: ["get", "create", "update"] + - apiGroups: ["cns.vmware.com"] + resources: ["cnsvolumeoperationrequests"] + verbs: ["create", "get", "list", "update", "delete"] + - apiGroups: [ "cns.vmware.com" ] + resources: [ "csinodetopologies" ] + verbs: ["get", "update", "watch", "list"] + - apiGroups: ["storage.k8s.io"] + resources: ["volumeattachments/status"] + verbs: ["patch"] + - apiGroups: [ "snapshot.storage.k8s.io" ] + resources: [ "volumesnapshots" ] + verbs: [ "get", "list" ] + - apiGroups: [ "snapshot.storage.k8s.io" ] + resources: [ "volumesnapshotclasses" ] + verbs: [ "watch", "get", "list" ] + - apiGroups: [ "snapshot.storage.k8s.io" ] + resources: [ "volumesnapshotcontents" ] + verbs: [ "create", "get", "list", "watch", "update", "delete", "patch" ] + - apiGroups: [ "snapshot.storage.k8s.io" ] + resources: [ "volumesnapshotcontents/status" ] + verbs: [ "update", "patch" ] +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: vsphere-csi-controller-binding +subjects: + - kind: ServiceAccount + name: vsphere-csi-controller + namespace: "{{ vsphere_csi_namespace }}" +roleRef: + kind: ClusterRole + name: vsphere-csi-controller-role + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-service.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-service.yml.j2 new file mode 100644 index 0000000..75967ba --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-controller-service.yml.j2 @@ -0,0 +1,19 @@ +apiVersion: v1 +kind: Service +metadata: + name: vsphere-csi-controller + namespace: "{{ vsphere_csi_namespace }}" + labels: + app: vsphere-csi-controller +spec: + ports: + - name: ctlr + port: 2112 + targetPort: 2112 + protocol: TCP + - name: syncer + port: 2113 + targetPort: 2113 + protocol: TCP + selector: + app: vsphere-csi-controller diff --git a/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-driver.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-driver.yml.j2 new file mode 100644 index 0000000..ad3260e --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-driver.yml.j2 @@ -0,0 +1,7 @@ +apiVersion: storage.k8s.io/v1 +kind: CSIDriver +metadata: + name: csi.vsphere.vmware.com +spec: + attachRequired: true + podInfoOnMount: false diff --git a/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-namespace.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-namespace.yml.j2 new file mode 100644 index 0000000..6cf3150 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-namespace.yml.j2 @@ -0,0 +1,4 @@ +apiVersion: v1 +kind: Namespace +metadata: + name: "{{ vsphere_csi_namespace }}" diff --git a/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-node-rbac.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-node-rbac.yml.j2 new file mode 100644 index 0000000..42896e1 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-node-rbac.yml.j2 @@ -0,0 +1,55 @@ +--- +kind: ServiceAccount +apiVersion: v1 +metadata: + name: vsphere-csi-node + namespace: "{{ vsphere_csi_namespace }}" +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: vsphere-csi-node-cluster-role +rules: + - apiGroups: ["cns.vmware.com"] + resources: ["csinodetopologies"] + verbs: ["create", "watch", "get", "patch" ] + - apiGroups: [""] + resources: ["nodes"] + verbs: ["get"] +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: vsphere-csi-node-cluster-role-binding +subjects: + - kind: ServiceAccount + name: vsphere-csi-node + namespace: "{{ vsphere_csi_namespace }}" +roleRef: + kind: ClusterRole + name: vsphere-csi-node-cluster-role + apiGroup: rbac.authorization.k8s.io +--- +kind: Role +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: vsphere-csi-node-role + namespace: "{{ vsphere_csi_namespace }}" +rules: + - apiGroups: [""] + resources: ["configmaps"] + verbs: ["get", "list", "watch"] +--- +kind: RoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: vsphere-csi-node-binding + namespace: "{{ vsphere_csi_namespace }}" +subjects: + - kind: ServiceAccount + name: vsphere-csi-node + namespace: "{{ vsphere_csi_namespace }}" +roleRef: + kind: Role + name: vsphere-csi-node-role + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-node.yml.j2 b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-node.yml.j2 new file mode 100644 index 0000000..e110ee3 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/csi_driver/vsphere/templates/vsphere-csi-node.yml.j2 @@ -0,0 +1,170 @@ +kind: DaemonSet +apiVersion: apps/v1 +metadata: + name: vsphere-csi-node + namespace: "{{ vsphere_csi_namespace }}" +spec: + selector: + matchLabels: + app: vsphere-csi-node + updateStrategy: + type: "RollingUpdate" + rollingUpdate: + maxUnavailable: 1 + template: + metadata: + labels: + app: vsphere-csi-node + role: vsphere-csi + spec: + priorityClassName: system-node-critical + nodeSelector: + kubernetes.io/os: linux +{% if vsphere_csi_node_affinity %} + affinity: + {{ vsphere_csi_node_affinity | to_nice_yaml | indent(width=8) }} +{% endif %} + serviceAccountName: vsphere-csi-node + hostNetwork: true + dnsPolicy: "ClusterFirstWithHostNet" + containers: + - name: node-driver-registrar + image: {{ kube_image_repo }}/sig-storage/csi-node-driver-registrar:{{ vsphere_csi_node_driver_registrar_image_tag }} +{% if external_vsphere_version < "7.0u1" %} + lifecycle: + preStop: + exec: + command: ["/bin/sh", "-c", "rm -rf /registration/csi.vsphere.vmware.com-reg.sock /csi/csi.sock"] +{% endif %} + args: + - "--v=5" + - "--csi-address=$(ADDRESS)" + - "--kubelet-registration-path=$(DRIVER_REG_SOCK_PATH)" +{% if vsphere_csi_node_driver_registrar_resources | length > 0 %} + resources: + {{ vsphere_csi_node_driver_registrar_resources | default({}) | to_nice_yaml | trim | indent(width=10) }} +{% endif %} + env: + - name: ADDRESS + value: /csi/csi.sock + - name: DRIVER_REG_SOCK_PATH + value: /var/lib/kubelet/plugins/csi.vsphere.vmware.com/csi.sock + volumeMounts: + - name: plugin-dir + mountPath: /csi + - name: registration-dir + mountPath: /registration + livenessProbe: + exec: + command: + - /csi-node-driver-registrar + - --kubelet-registration-path=/var/lib/kubelet/plugins/csi.vsphere.vmware.com/csi.sock + - --mode=kubelet-registration-probe + initialDelaySeconds: 3 + - name: vsphere-csi-node + image: {{ gcr_image_repo }}/cloud-provider-vsphere/csi/release/driver:{{ vsphere_csi_driver_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - "--fss-name=internal-feature-states.csi.vsphere.vmware.com" + - "--fss-namespace={{ vsphere_csi_namespace }}" + imagePullPolicy: "Always" +{% if vsphere_csi_driver_resources | length > 0 %} + resources: + {{ vsphere_csi_driver_resources | default({}) | to_nice_yaml | trim | indent(width=10) }} +{% endif %} + env: + - name: NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: CSI_ENDPOINT + value: unix:///csi/csi.sock + - name: MAX_VOLUMES_PER_NODE + value: "59" # Maximum number of volumes that controller can publish to the node. If value is not set or zero Kubernetes decide how many volumes can be published by the controller to the node. + - name: X_CSI_MODE + value: "node" + - name: X_CSI_SPEC_REQ_VALIDATION + value: "false" + - name: X_CSI_DEBUG + value: "true" + - name: X_CSI_SPEC_DISABLE_LEN_CHECK + value: "true" + - name: LOGGER_LEVEL + value: "PRODUCTION" # Options: DEVELOPMENT, PRODUCTION + - name: GODEBUG + value: x509sha1=1 + - name: NODEGETINFO_WATCH_TIMEOUT_MINUTES + value: "1" + securityContext: + privileged: true + capabilities: + add: ["SYS_ADMIN"] + allowPrivilegeEscalation: true + volumeMounts: + - name: plugin-dir + mountPath: /csi + - name: pods-mount-dir + mountPath: /var/lib/kubelet + # needed so that any mounts setup inside this container are + # propagated back to the host machine. + mountPropagation: "Bidirectional" + - name: device-dir + mountPath: /dev + - name: blocks-dir + mountPath: /sys/block + - name: sys-devices-dir + mountPath: /sys/devices + ports: + - containerPort: 9808 + name: healthz + livenessProbe: + httpGet: + path: /healthz + port: healthz + initialDelaySeconds: 10 + timeoutSeconds: 5 + periodSeconds: 5 + failureThreshold: 3 + - name: liveness-probe + image: {{ kube_image_repo }}/sig-storage/livenessprobe:{{ vsphere_csi_liveness_probe_image_tag }} + args: +{% if external_vsphere_version >= "7.0u1" %} + - "--v=4" +{% endif %} + - "--csi-address=/csi/csi.sock" +{% if vsphere_csi_liveness_probe_ds_resources | length > 0 %} + resources: + {{ vsphere_csi_liveness_probe_ds_resources | default({}) | to_nice_yaml | trim | indent(width=10) }} +{% endif %} + volumeMounts: + - name: plugin-dir + mountPath: /csi + volumes: + - name: registration-dir + hostPath: + path: /var/lib/kubelet/plugins_registry + type: Directory + - name: plugin-dir + hostPath: + path: /var/lib/kubelet/plugins/csi.vsphere.vmware.com + type: DirectoryOrCreate + - name: pods-mount-dir + hostPath: + path: /var/lib/kubelet + type: Directory + - name: device-dir + hostPath: + path: /dev + - name: blocks-dir + hostPath: + path: /sys/block + type: Directory + - name: sys-devices-dir + hostPath: + path: /sys/devices + type: Directory + tolerations: + - effect: NoExecute + operator: Exists + - effect: NoSchedule + operator: Exists diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/defaults/main.yml b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/defaults/main.yml new file mode 100644 index 0000000..5d9ba29 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/defaults/main.yml @@ -0,0 +1,14 @@ +--- +external_hcloud_cloud: + hcloud_api_token: "" + token_secret_name: hcloud + + service_account_name: cloud-controller-manager + + controller_image_tag: "latest" + ## A dictionary of extra arguments to add to the openstack cloud controller manager daemonset + ## Format: + ## external_hcloud_cloud.controller_extra_args: + ## arg1: "value1" + ## arg2: "value2" + controller_extra_args: {} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/tasks/main.yml b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/tasks/main.yml new file mode 100644 index 0000000..c626e78 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/tasks/main.yml @@ -0,0 +1,30 @@ +--- +- name: External Hcloud Cloud Controller | Generate Manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + group: "{{ kube_cert_group }}" + mode: 0640 + with_items: + - {name: external-hcloud-cloud-secret, file: external-hcloud-cloud-secret.yml} + - {name: external-hcloud-cloud-service-account, file: external-hcloud-cloud-service-account.yml} + - {name: external-hcloud-cloud-role-bindings, file: external-hcloud-cloud-role-bindings.yml} + - {name: "{{ 'external-hcloud-cloud-controller-manager-ds-with-networks' if external_hcloud_cloud.with_networks else 'external-hcloud-cloud-controller-manager-ds' }}", file: "{{ 'external-hcloud-cloud-controller-manager-ds-with-networks.yml' if external_hcloud_cloud.with_networks else 'external-hcloud-cloud-controller-manager-ds.yml' }}"} + + register: external_hcloud_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + tags: external-hcloud + +- name: External Hcloud Cloud Controller | Apply Manifests + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ external_hcloud_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item.file }}" + tags: external-hcloud diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-controller-manager-ds-with-networks.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-controller-manager-ds-with-networks.yml.j2 new file mode 100644 index 0000000..ec64d9a --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-controller-manager-ds-with-networks.yml.j2 @@ -0,0 +1,96 @@ +--- +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: hcloud-cloud-controller-manager + namespace: kube-system + labels: + k8s-app: hcloud-cloud-controller-manger +spec: + selector: + matchLabels: + app: hcloud-cloud-controller-manager + template: + metadata: + labels: + app: hcloud-cloud-controller-manager + annotations: + scheduler.alpha.kubernetes.io/critical-pod: '' + spec: + serviceAccountName: {{ external_hcloud_cloud.service_account_name }} + dnsPolicy: Default + tolerations: + - key: "node.cloudprovider.kubernetes.io/uninitialized" + value: "true" + effect: "NoSchedule" + - key: "CriticalAddonsOnly" + operator: "Exists" + - key: "node-role.kubernetes.io/master" + effect: NoSchedule + operator: Exists + - key: "node-role.kubernetes.io/control-plane" + effect: NoSchedule + operator: Exists + - key: "node.kubernetes.io/not-ready" + effect: "NoSchedule" + hostNetwork: true + containers: + - image: {{ docker_image_repo }}/hetznercloud/hcloud-cloud-controller-manager:{{ external_hcloud_cloud.controller_image_tag }} + name: hcloud-cloud-controller-manager + command: + - "/bin/hcloud-cloud-controller-manager" + - "--cloud-provider=hcloud" + - "--leader-elect=false" + - "--allow-untagged-cloud" + - "--allocate-node-cidrs=true" + - "--cluster-cidr={{ kube_pods_subnet }}" +{% if external_hcloud_cloud.controller_extra_args is defined %} + + args: +{% for key, value in external_hcloud_cloud.controller_extra_args.items() %} + - "{{ '--' + key + '=' + value }}" +{% endfor %} +{% endif %} + resources: + requests: + cpu: 100m + memory: 50Mi + env: + - name: NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: HCLOUD_TOKEN + valueFrom: + secretKeyRef: + name: {{ external_hcloud_cloud.token_secret_name }} + key: token + - name: HCLOUD_NETWORK + valueFrom: + secretKeyRef: + name: {{ external_hcloud_cloud.token_secret_name }} + key: network +{% if external_hcloud_cloud.network_routes_enabled is defined %} + - name: HCLOUD_NETWORK_ROUTES_ENABLED + value: "{{ external_hcloud_cloud.network_routes_enabled }}" +{% endif %} +{% if external_hcloud_cloud.load_balancers_location is defined %} + - name: HCLOUD_LOAD_BALANCERS_LOCATION + value: "{{ external_hcloud_cloud.load_balancers_location }}" +{% endif %} +{% if external_hcloud_cloud.load_balancers_network_zone is defined %} + - name: HCLOUD_LOAD_BALANCERS_NETWORK_ZONE + value: "{{ external_hcloud_cloud.load_balancers_network_zone }}" +{% endif %} +{% if external_hcloud_cloud.load_balancers_disable_private_ingress is defined %} + - name: HCLOUD_LOAD_BALANCERS_DISABLE_PRIVATE_INGRESS + value: "{{ external_hcloud_cloud.load_balancers_disable_private_ingress }}" +{% endif %} +{% if external_hcloud_cloud.load_balancers_use_private_ip is defined %} + - name: HCLOUD_LOAD_BALANCERS_USE_PRIVATE_IP + value: "{{ external_hcloud_cloud.load_balancers_use_private_ip }}" +{% endif %} +{% if external_hcloud_cloud.load_balancers_enabled is defined %} + - name: HCLOUD_LOAD_BALANCERS_ENABLED + value: "{{ external_hcloud_cloud.load_balancers_enabled }}" +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-controller-manager-ds.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-controller-manager-ds.yml.j2 new file mode 100644 index 0000000..a581781 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-controller-manager-ds.yml.j2 @@ -0,0 +1,94 @@ +--- +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: hcloud-cloud-controller-manager + namespace: kube-system + labels: + k8s-app: hcloud-cloud-controller-manger +spec: + selector: + matchLabels: + app: hcloud-cloud-controller-manager + updateStrategy: + type: RollingUpdate + template: + metadata: + labels: + app: hcloud-cloud-controller-manager + annotations: + scheduler.alpha.kubernetes.io/critical-pod: '' + spec: + serviceAccountName: {{ external_hcloud_cloud.service_account_name }} + dnsPolicy: Default + tolerations: + - key: "node.cloudprovider.kubernetes.io/uninitialized" + value: "true" + effect: "NoSchedule" + - key: "CriticalAddonsOnly" + operator: "Exists" + - key: "node-role.kubernetes.io/master" + effect: NoSchedule + - key: "node-role.kubernetes.io/control-plane" + effect: NoSchedule + - key: "node.kubernetes.io/not-ready" + effect: "NoSchedule" + containers: + - image: {{ docker_image_repo }}/hetznercloud/hcloud-cloud-controller-manager:{{ external_hcloud_cloud.controller_image_tag }} + name: hcloud-cloud-controller-manager + command: + - "/bin/hcloud-cloud-controller-manager" + - "--cloud-provider=hcloud" + - "--leader-elect=false" + - "--allow-untagged-cloud" +{% if external_hcloud_cloud.controller_extra_args is defined %} + args: +{% for key, value in external_hcloud_cloud.controller_extra_args.items() %} + - "{{ '--' + key + '=' + value }}" +{% endfor %} +{% endif %} + resources: + requests: + cpu: 100m + memory: 50Mi + env: + - name: NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: HCLOUD_TOKEN + valueFrom: + secretKeyRef: + name: {{ external_hcloud_cloud.token_secret_name }} + key: token +{% if external_hcloud_cloud.network_name is defined %} + - name: HCLOUD_NETWORK + valueFrom: + secretKeyRef: + name: {{ external_hcloud_cloud.token_secret_name }} + key: network +{% endif %} +{% if external_hcloud_cloud.network_routes_enabled is defined %} + - name: HCLOUD_NETWORK_ROUTES_ENABLED + value: "{{ external_hcloud_cloud.network_routes_enabled }}" +{% endif %} +{% if external_hcloud_cloud.load_balancers_location is defined %} + - name: HCLOUD_LOAD_BALANCERS_LOCATION + value: "{{ external_hcloud_cloud.load_balancers_location }}" +{% endif %} +{% if external_hcloud_cloud.load_balancers_network_zone is defined %} + - name: HCLOUD_LOAD_BALANCERS_NETWORK_ZONE + value: "{{ external_hcloud_cloud.load_balancers_network_zone }}" +{% endif %} +{% if external_hcloud_cloud.load_balancers_disable_private_ingress is defined %} + - name: HCLOUD_LOAD_BALANCERS_DISABLE_PRIVATE_INGRESS + value: "{{ external_hcloud_cloud.load_balancers_disable_private_ingress }}" +{% endif %} +{% if external_hcloud_cloud.load_balancers_use_private_ip is defined %} + - name: HCLOUD_LOAD_BALANCERS_USE_PRIVATE_IP + value: "{{ external_hcloud_cloud.load_balancers_use_private_ip }}" +{% endif %} +{% if external_hcloud_cloud.load_balancers_enabled is defined %} + - name: HCLOUD_LOAD_BALANCERS_ENABLED + value: "{{ external_hcloud_cloud.load_balancers_enabled }}" +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-role-bindings.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-role-bindings.yml.j2 new file mode 100644 index 0000000..270c947 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-role-bindings.yml.j2 @@ -0,0 +1,13 @@ +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: system:cloud-controller-manager +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cluster-admin +subjects: + - kind: ServiceAccount + name: {{ external_hcloud_cloud.service_account_name }} + namespace: kube-system diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-secret.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-secret.yml.j2 new file mode 100644 index 0000000..ab3df74 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-secret.yml.j2 @@ -0,0 +1,15 @@ +--- +apiVersion: v1 +kind: Secret +metadata: + name: "{{ external_hcloud_cloud.token_secret_name }}" + namespace: kube-system +data: + token: "{{ external_hcloud_cloud.hcloud_api_token | b64encode }}" +{% if external_hcloud_cloud.with_networks or external_hcloud_cloud.network_name is defined %} +{% if network_id is defined%} + network: "{{ network_id | b64encode }}" +{% else %} + network: "{{ external_hcloud_cloud.network_name | b64encode }}" +{% endif %} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-service-account.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-service-account.yml.j2 new file mode 100644 index 0000000..93277dd --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/hcloud/templates/external-hcloud-cloud-service-account.yml.j2 @@ -0,0 +1,6 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ external_hcloud_cloud.service_account_name }} + namespace: kube-system diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/defaults/main.yml b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/defaults/main.yml new file mode 100644 index 0000000..6d89c57 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/defaults/main.yml @@ -0,0 +1,19 @@ +--- +# The external cloud controller will need credentials to access +# openstack apis. Per default these values will be +# read from the environment. +external_huaweicloud_auth_url: "{{ lookup('env','OS_AUTH_URL') }}" +external_huaweicloud_access_key: "{{ lookup('env','OS_ACCESS_KEY') }}" +external_huaweicloud_secret_key: "{{ lookup('env','OS_SECRET_KEY') }}" +external_huaweicloud_region: "{{ lookup('env','OS_REGION_NAME') }}" +external_huaweicloud_project_id: "{{ lookup('env','OS_TENANT_ID')| default(lookup('env','OS_PROJECT_ID'),true) }}" +external_huaweicloud_cloud: "{{ lookup('env','OS_CLOUD') }}" + +## A dictionary of extra arguments to add to the huawei cloud controller manager deployment +## Format: +## external_huawei_cloud_controller_extra_args: +## arg1: "value1" +## arg2: "value2" +external_huawei_cloud_controller_extra_args: {} +external_huawei_cloud_controller_image_repo: "swr.ap-southeast-1.myhuaweicloud.com" +external_huawei_cloud_controller_image_tag: "v0.26.3" diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/tasks/huaweicloud-credential-check.yml b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/tasks/huaweicloud-credential-check.yml new file mode 100644 index 0000000..79172ff --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/tasks/huaweicloud-credential-check.yml @@ -0,0 +1,33 @@ +--- +- name: External Huawei Cloud Controller | check external_huaweicloud_auth_url value + fail: + msg: "external_huaweicloud_auth_url is missing" + when: external_huaweicloud_auth_url is not defined or not external_huaweicloud_auth_url + + +- name: External Huawei Cloud Controller | check external_huaweicloud_access_key value + fail: + msg: "you must set external_huaweicloud_access_key" + when: + - external_huaweicloud_access_key is not defined or not external_huaweicloud_access_key + +- name: External Huawei Cloud Controller | check external_huaweicloud_secret_key value + fail: + msg: "external_huaweicloud_secret_key is missing" + when: + - external_huaweicloud_access_key is defined + - external_huaweicloud_access_key|length > 0 + - external_huaweicloud_secret_key is not defined or not external_huaweicloud_secret_key + + +- name: External Huawei Cloud Controller | check external_huaweicloud_region value + fail: + msg: "external_huaweicloud_region is missing" + when: external_huaweicloud_region is not defined or not external_huaweicloud_region + + +- name: External Huawei Cloud Controller | check external_huaweicloud_project_id value + fail: + msg: "one of external_huaweicloud_project_id must be specified" + when: + - external_huaweicloud_project_id is not defined or not external_huaweicloud_project_id diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/tasks/main.yml b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/tasks/main.yml new file mode 100644 index 0000000..880be0d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/tasks/main.yml @@ -0,0 +1,49 @@ +--- +- name: External Huawei Cloud Controller | Check Huawei credentials + include_tasks: huaweicloud-credential-check.yml + tags: external-huaweicloud + +- name: External huaweicloud Cloud Controller | Get base64 cacert + slurp: + src: "{{ external_huaweicloud_cacert }}" + register: external_huaweicloud_cacert_b64 + when: + - inventory_hostname == groups['kube_control_plane'][0] + - external_huaweicloud_cacert is defined + - external_huaweicloud_cacert | length > 0 + tags: external-huaweicloud + +- name: External huaweicloud Cloud Controller | Get base64 cloud-config + set_fact: + external_huawei_cloud_config_secret: "{{ lookup('template', 'external-huawei-cloud-config.j2') | b64encode }}" + when: inventory_hostname == groups['kube_control_plane'][0] + tags: external-huaweicloud + +- name: External Huawei Cloud Controller | Generate Manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + group: "{{ kube_cert_group }}" + mode: 0640 + with_items: + - {name: external-huawei-cloud-config-secret, file: external-huawei-cloud-config-secret.yml} + - {name: external-huawei-cloud-controller-manager-roles, file: external-huawei-cloud-controller-manager-roles.yml} + - {name: external-huawei-cloud-controller-manager-role-bindings, file: external-huawei-cloud-controller-manager-role-bindings.yml} + - {name: external-huawei-cloud-controller-manager-ds, file: external-huawei-cloud-controller-manager-ds.yml} + register: external_huaweicloud_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + tags: external-huaweicloud + +- name: External Huawei Cloud Controller | Apply Manifests + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ external_huaweicloud_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item.file }}" + tags: external-huaweicloud diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-config-secret.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-config-secret.yml.j2 new file mode 100644 index 0000000..1f0bbf3 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-config-secret.yml.j2 @@ -0,0 +1,10 @@ +# This YAML file contains secret objects, +# which are necessary to run external huaweicloud cloud controller. + +kind: Secret +apiVersion: v1 +metadata: + name: external-huawei-cloud-config + namespace: kube-system +data: + cloud-config: {{ external_huawei_cloud_config_secret }} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-config.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-config.j2 new file mode 100644 index 0000000..07f1771 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-config.j2 @@ -0,0 +1,23 @@ +[Global] +auth-url="{{ external_huaweicloud_auth_url }}" +{% if external_huaweicloud_access_key is defined and external_huaweicloud_access_key != "" %} +access-key={{ external_huaweicloud_access_key }} +{% endif %} +{% if external_huaweicloud_secret_key is defined and external_huaweicloud_secret_key != "" %} +secret-key={{ external_huaweicloud_secret_key }} +{% endif %} +region="{{ external_huaweicloud_region }}" +{% if external_huaweicloud_project_id is defined and external_huaweicloud_project_id != "" %} +project-id="{{ external_huaweicloud_project_id }}" +{% endif %} +{% if external_huaweicloud_cloud is defined and external_huaweicloud_cloud != "" %} +cloud="{{ external_huaweicloud_cloud }}" +{% endif %} + +[VPC] +{% if external_huaweicloud_lbaas_subnet_id is defined %} +subnet-id={{ external_huaweicloud_lbaas_subnet_id }} +{% endif %} +{% if external_huaweicloud_lbaas_network_id is defined %} +id={{ external_huaweicloud_lbaas_network_id }} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-controller-manager-ds.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-controller-manager-ds.yml.j2 new file mode 100644 index 0000000..5e4b424 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-controller-manager-ds.yml.j2 @@ -0,0 +1,93 @@ +kind: Namespace +apiVersion: v1 +metadata: + name: huawei-cloud-provider +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: cloud-controller-manager + namespace: kube-system +--- +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: huawei-cloud-controller-manager + namespace: kube-system + labels: + k8s-app: huawei-cloud-controller-manager +spec: + selector: + matchLabels: + k8s-app: huawei-cloud-controller-manager + updateStrategy: + type: RollingUpdate + template: + metadata: + labels: + k8s-app: huawei-cloud-controller-manager + spec: + nodeSelector: + node-role.kubernetes.io/control-plane: "" + securityContext: + runAsUser: 1001 + tolerations: + - key: node.cloudprovider.kubernetes.io/uninitialized + value: "true" + effect: NoSchedule + - key: node-role.kubernetes.io/master + effect: NoSchedule + - key: node-role.kubernetes.io/control-plane + effect: NoSchedule + serviceAccountName: cloud-controller-manager + containers: + - name: huawei-cloud-controller-manager + image: {{ external_huawei_cloud_controller_image_repo }}/k8s-cloudprovider/huawei-cloud-controller-manager:{{ external_huawei_cloud_controller_image_tag }} + args: + - /bin/huawei-cloud-controller-manager + - --v=1 + - --cloud-config=$(CLOUD_CONFIG) + - --cloud-provider=huaweicloud + - --use-service-account-credentials=true +{% for key, value in external_huawei_cloud_controller_extra_args.items() %} + - "{{ '--' + key + '=' + value }}" +{% endfor %} + volumeMounts: + - mountPath: /etc/kubernetes + name: k8s-certs + readOnly: true + - mountPath: /etc/ssl/certs + name: ca-certs + readOnly: true + - mountPath: /etc/config + name: cloud-config-volume + readOnly: true +{% if kubelet_flexvolumes_plugins_dir is defined %} + - mountPath: /usr/libexec/kubernetes/kubelet-plugins/volume/exec + name: flexvolume-dir +{% endif %} + resources: + requests: + cpu: 200m + env: + - name: CLOUD_CONFIG + value: /etc/config/cloud-config + hostNetwork: true + volumes: +{% if kubelet_flexvolumes_plugins_dir is defined %} + - name: flexvolume-dir + hostPath: + path: "{{ kubelet_flexvolumes_plugins_dir }}" + type: DirectoryOrCreate +{% endif %} + - name: k8s-certs + hostPath: + path: /etc/kubernetes + type: DirectoryOrCreate + - name: ca-certs + hostPath: + path: /etc/ssl/certs + type: DirectoryOrCreate + - name: cloud-config-volume + secret: + secretName: external-huawei-cloud-config diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-controller-manager-role-bindings.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-controller-manager-role-bindings.yml.j2 new file mode 100644 index 0000000..bbdf336 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-controller-manager-role-bindings.yml.j2 @@ -0,0 +1,16 @@ +apiVersion: v1 +items: +- apiVersion: rbac.authorization.k8s.io/v1 + kind: ClusterRoleBinding + metadata: + name: system:cloud-controller-manager + roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: system:cloud-controller-manager + subjects: + - kind: ServiceAccount + name: cloud-controller-manager + namespace: kube-system +kind: List +metadata: {} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-controller-manager-roles.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-controller-manager-roles.yml.j2 new file mode 100644 index 0000000..2e2d8b6 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/huaweicloud/templates/external-huawei-cloud-controller-manager-roles.yml.j2 @@ -0,0 +1,117 @@ +apiVersion: v1 +items: +- apiVersion: rbac.authorization.k8s.io/v1 + kind: ClusterRole + metadata: + name: system:cloud-controller-manager + rules: + - resources: + - tokenreviews + verbs: + - get + - list + - watch + - create + - update + - patch + apiGroups: + - authentication.k8s.io + - resources: + - configmaps + - endpoints + - pods + - services + - secrets + - serviceaccounts + - serviceaccounts/token + verbs: + - get + - list + - watch + - create + - update + - patch + apiGroups: + - '' + - resources: + - nodes + verbs: + - get + - list + - watch + - delete + - patch + - update + apiGroups: + - '' + - resources: + - services/status + - pods/status + verbs: + - update + - patch + apiGroups: + - '' + - resources: + - nodes/status + verbs: + - patch + - update + apiGroups: + - '' + - resources: + - events + - endpoints + verbs: + - create + - patch + - update + apiGroups: + - '' + - resources: + - leases + verbs: + - get + - update + - create + - delete + apiGroups: + - coordination.k8s.io + - resources: + - customresourcedefinitions + verbs: + - get + - update + - create + - delete + apiGroups: + - apiextensions.k8s.io + - resources: + - ingresses + verbs: + - get + - list + - watch + - update + - create + - patch + - delete + apiGroups: + - networking.k8s.io + - resources: + - ingresses/status + verbs: + - update + - patch + apiGroups: + - networking.k8s.io + - resources: + - endpointslices + verbs: + - get + - list + - watch + apiGroups: + - discovery.k8s.io +kind: List +metadata: {} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/meta/main.yml b/kubespray/roles/kubernetes-apps/external_cloud_controller/meta/main.yml new file mode 100644 index 0000000..b1fc4ad --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/meta/main.yml @@ -0,0 +1,42 @@ +--- +dependencies: + - role: kubernetes-apps/external_cloud_controller/openstack + when: + - cloud_provider is defined + - cloud_provider == "external" + - external_cloud_provider is defined + - external_cloud_provider == "openstack" + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - external-cloud-controller + - external-openstack + - role: kubernetes-apps/external_cloud_controller/vsphere + when: + - cloud_provider is defined + - cloud_provider == "external" + - external_cloud_provider is defined + - external_cloud_provider == "vsphere" + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - external-cloud-controller + - external-vsphere + - role: kubernetes-apps/external_cloud_controller/hcloud + when: + - cloud_provider is defined + - cloud_provider == "external" + - external_cloud_provider is defined + - external_cloud_provider == "hcloud" + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - external-cloud-controller + - external-hcloud + - role: kubernetes-apps/external_cloud_controller/huaweicloud + when: + - cloud_provider is defined + - cloud_provider == "external" + - external_cloud_provider is defined + - external_cloud_provider == "huaweicloud" + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - external-cloud-controller + - external-huaweicloud diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/OWNERS b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/OWNERS new file mode 100644 index 0000000..6cfbaa8 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/OWNERS @@ -0,0 +1,6 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +approvers: +reviewers: + - alijahnas + - luckySB diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/defaults/main.yml b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/defaults/main.yml new file mode 100644 index 0000000..4bcf135 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/defaults/main.yml @@ -0,0 +1,25 @@ +--- +# The external cloud controller will need credentials to access +# openstack apis. Per default these values will be +# read from the environment. +external_openstack_auth_url: "{{ lookup('env', 'OS_AUTH_URL') }}" +external_openstack_username: "{{ lookup('env', 'OS_USERNAME') }}" +external_openstack_password: "{{ lookup('env', 'OS_PASSWORD') }}" +external_openstack_application_credential_id: "{{ lookup('env', 'OS_APPLICATION_CREDENTIAL_ID') }}" +external_openstack_application_credential_name: "{{ lookup('env', 'OS_APPLICATION_CREDENTIAL_NAME') }}" +external_openstack_application_credential_secret: "{{ lookup('env', 'OS_APPLICATION_CREDENTIAL_SECRET') }}" +external_openstack_region: "{{ lookup('env', 'OS_REGION_NAME') }}" +external_openstack_tenant_id: "{{ lookup('env', 'OS_TENANT_ID') | default(lookup('env', 'OS_PROJECT_ID'), true) }}" +external_openstack_tenant_name: "{{ lookup('env', 'OS_TENANT_NAME') | default(lookup('env', 'OS_PROJECT_NAME'), true) }}" +external_openstack_domain_name: "{{ lookup('env', 'OS_USER_DOMAIN_NAME') }}" +external_openstack_domain_id: "{{ lookup('env', 'OS_USER_DOMAIN_ID') }}" +external_openstack_cacert: "{{ lookup('env', 'OS_CACERT') }}" + +## A dictionary of extra arguments to add to the openstack cloud controller manager daemonset +## Format: +## external_openstack_cloud_controller_extra_args: +## arg1: "value1" +## arg2: "value2" +external_openstack_cloud_controller_extra_args: {} +external_openstack_cloud_controller_image_tag: "v1.25.3" +external_openstack_cloud_controller_bind_address: 127.0.0.1 diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/tasks/main.yml b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/tasks/main.yml new file mode 100644 index 0000000..787dbb4 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/tasks/main.yml @@ -0,0 +1,49 @@ +--- +- name: External OpenStack Cloud Controller | Check OpenStack credentials + include_tasks: openstack-credential-check.yml + tags: external-openstack + +- name: External OpenStack Cloud Controller | Get base64 cacert + slurp: + src: "{{ external_openstack_cacert }}" + register: external_openstack_cacert_b64 + when: + - inventory_hostname == groups['kube_control_plane'][0] + - external_openstack_cacert is defined + - external_openstack_cacert | length > 0 + tags: external-openstack + +- name: External OpenStack Cloud Controller | Get base64 cloud-config + set_fact: + external_openstack_cloud_config_secret: "{{ lookup('template', 'external-openstack-cloud-config.j2') | b64encode }}" + when: inventory_hostname == groups['kube_control_plane'][0] + tags: external-openstack + +- name: External OpenStack Cloud Controller | Generate Manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + group: "{{ kube_cert_group }}" + mode: 0640 + with_items: + - {name: external-openstack-cloud-config-secret, file: external-openstack-cloud-config-secret.yml} + - {name: external-openstack-cloud-controller-manager-roles, file: external-openstack-cloud-controller-manager-roles.yml} + - {name: external-openstack-cloud-controller-manager-role-bindings, file: external-openstack-cloud-controller-manager-role-bindings.yml} + - {name: external-openstack-cloud-controller-manager-ds, file: external-openstack-cloud-controller-manager-ds.yml} + register: external_openstack_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + tags: external-openstack + +- name: External OpenStack Cloud Controller | Apply Manifests + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ external_openstack_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item.file }}" + tags: external-openstack diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/tasks/openstack-credential-check.yml b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/tasks/openstack-credential-check.yml new file mode 100644 index 0000000..6a14658 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/tasks/openstack-credential-check.yml @@ -0,0 +1,66 @@ +--- +- name: External OpenStack Cloud Controller | check external_openstack_auth_url value + fail: + msg: "external_openstack_auth_url is missing" + when: external_openstack_auth_url is not defined or not external_openstack_auth_url + + +- name: External OpenStack Cloud Controller | check external_openstack_username or external_openstack_application_credential_name value + fail: + msg: "you must either set external_openstack_username or external_openstack_application_credential_name" + when: + - external_openstack_username is not defined or not external_openstack_username + - external_openstack_application_credential_name is not defined or not external_openstack_application_credential_name + + +- name: External OpenStack Cloud Controller | check external_openstack_application_credential_id value + fail: + msg: "external_openstack_application_credential_id is missing" + when: + - external_openstack_application_credential_name is defined + - external_openstack_application_credential_name | length > 0 + - external_openstack_application_credential_id is not defined or not external_openstack_application_credential_id + + +- name: External OpenStack Cloud Controller | check external_openstack_application_credential_secret value + fail: + msg: "external_openstack_application_credential_secret is missing" + when: + - external_openstack_application_credential_name is defined + - external_openstack_application_credential_name | length > 0 + - external_openstack_application_credential_secret is not defined or not external_openstack_application_credential_secret + + +- name: External OpenStack Cloud Controller | check external_openstack_password value + fail: + msg: "external_openstack_password is missing" + when: + - external_openstack_username is defined + - external_openstack_username | length > 0 + - external_openstack_application_credential_name is not defined or not external_openstack_application_credential_name + - external_openstack_application_credential_secret is not defined or not external_openstack_application_credential_secret + - external_openstack_password is not defined or not external_openstack_password + + +- name: External OpenStack Cloud Controller | check external_openstack_region value + fail: + msg: "external_openstack_region is missing" + when: external_openstack_region is not defined or not external_openstack_region + + +- name: External OpenStack Cloud Controller | check external_openstack_tenant_id value + fail: + msg: "one of external_openstack_tenant_id or external_openstack_tenant_name must be specified" + when: + - external_openstack_tenant_id is not defined or not external_openstack_tenant_id + - external_openstack_tenant_name is not defined or not external_openstack_tenant_name + - external_openstack_application_credential_name is not defined or not external_openstack_application_credential_name + + +- name: External OpenStack Cloud Controller | check external_openstack_domain_id value + fail: + msg: "one of external_openstack_domain_id or external_openstack_domain_name must be specified" + when: + - external_openstack_domain_id is not defined or not external_openstack_domain_id + - external_openstack_domain_name is not defined or not external_openstack_domain_name + - external_openstack_application_credential_name is not defined or not external_openstack_application_credential_name diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-config-secret.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-config-secret.yml.j2 new file mode 100644 index 0000000..2a6f6a8 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-config-secret.yml.j2 @@ -0,0 +1,13 @@ +# This YAML file contains secret objects, +# which are necessary to run external openstack cloud controller. + +kind: Secret +apiVersion: v1 +metadata: + name: external-openstack-cloud-config + namespace: kube-system +data: + cloud.conf: {{ external_openstack_cloud_config_secret }} +{% if external_openstack_cacert_b64.content is defined %} + ca.cert: {{ external_openstack_cacert_b64.content }} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-config.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-config.j2 new file mode 100644 index 0000000..08acd67 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-config.j2 @@ -0,0 +1,92 @@ +[Global] +auth-url="{{ external_openstack_auth_url }}" +{% if external_openstack_application_credential_id == "" and external_openstack_application_credential_name == "" %} +username="{{ external_openstack_username }}" +password="{{ external_openstack_password }}" +{% endif %} +{% if external_openstack_application_credential_id is defined and external_openstack_application_credential_id != "" %} +application-credential-id={{ external_openstack_application_credential_id }} +{% endif %} +{% if external_openstack_application_credential_name is defined and external_openstack_application_credential_name != "" %} +application-credential-name={{ external_openstack_application_credential_name }} +{% endif %} +{% if external_openstack_application_credential_secret is defined and external_openstack_application_credential_secret != "" %} +application-credential-secret={{ external_openstack_application_credential_secret }} +{% endif %} +region="{{ external_openstack_region }}" +{% if external_openstack_tenant_id is defined and external_openstack_tenant_id != "" %} +tenant-id="{{ external_openstack_tenant_id }}" +{% endif %} +{% if external_openstack_tenant_name is defined and external_openstack_tenant_name != "" %} +tenant-name="{{ external_openstack_tenant_name }}" +{% endif %} +{% if external_openstack_domain_name is defined and external_openstack_domain_name != "" %} +domain-name="{{ external_openstack_domain_name }}" +{% elif external_openstack_domain_id is defined and external_openstack_domain_id != "" %} +domain-id ="{{ external_openstack_domain_id }}" +{% endif %} +{% if external_openstack_cacert is defined and external_openstack_cacert != "" %} +ca-file="{{ kube_config_dir }}/external-openstack-cacert.pem" +{% endif %} + +[LoadBalancer] +enabled={{ external_openstack_lbaas_enabled | string | lower }} +{% if external_openstack_lbaas_floating_network_id is defined %} +floating-network-id={{ external_openstack_lbaas_floating_network_id }} +{% endif %} +{% if external_openstack_lbaas_floating_subnet_id is defined %} +floating-subnet-id={{ external_openstack_lbaas_floating_subnet_id }} +{% endif %} +{% if external_openstack_lbaas_method is defined %} +lb-method={{ external_openstack_lbaas_method }} +{% endif %} +{% if external_openstack_lbaas_provider is defined %} +lb-provider={{ external_openstack_lbaas_provider }} +{% endif %} +{% if external_openstack_lbaas_subnet_id is defined %} +subnet-id={{ external_openstack_lbaas_subnet_id }} +{% endif %} +{% if external_openstack_lbaas_network_id is defined %} +network-id={{ external_openstack_lbaas_network_id }} +{% endif %} +{% if external_openstack_lbaas_manage_security_groups is defined %} +manage-security-groups={{ external_openstack_lbaas_manage_security_groups }} +{% endif %} +{% if external_openstack_lbaas_create_monitor is defined %} +create-monitor={{ external_openstack_lbaas_create_monitor }} +{% endif %} +{% if external_openstack_lbaas_monitor_delay is defined %} +monitor-delay={{ external_openstack_lbaas_monitor_delay }} +{% endif %} +{% if external_openstack_lbaas_monitor_max_retries is defined %} +monitor-max-retries={{ external_openstack_lbaas_monitor_max_retries }} +{% endif %} +{% if external_openstack_lbaas_monitor_timeout is defined %} +monitor-timeout={{ external_openstack_lbaas_monitor_timeout }} +{% endif %} +{% if external_openstack_lbaas_internal_lb is defined %} +internal-lb={{ external_openstack_lbaas_internal_lb }} +{% endif %} +{% if external_openstack_enable_ingress_hostname is defined %} +enable-ingress-hostname={{ external_openstack_enable_ingress_hostname | string | lower }} +{% endif %} +{% if external_openstack_ingress_hostname_suffix is defined %} +ingress-hostname-suffix={{ external_openstack_ingress_hostname_suffix | string | lower }} +{% endif %} +{% if external_openstack_max_shared_lb is defined %} +max-shared-lb={{ external_openstack_max_shared_lb }} +{% endif %} + +[Networking] +ipv6-support-disabled={{ external_openstack_network_ipv6_disabled | string | lower }} +{% for network_name in external_openstack_network_internal_networks %} +internal-network-name="{{ network_name }}" +{% endfor %} +{% for network_name in external_openstack_network_public_networks %} +public-network-name="{{ network_name }}" +{% endfor %} + +[Metadata] +{% if external_openstack_metadata_search_order is defined %} +search-order="{{ external_openstack_metadata_search_order }}" +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-controller-manager-ds.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-controller-manager-ds.yml.j2 new file mode 100644 index 0000000..565875d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-controller-manager-ds.yml.j2 @@ -0,0 +1,111 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: cloud-controller-manager + namespace: kube-system +--- +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: openstack-cloud-controller-manager + namespace: kube-system + labels: + k8s-app: openstack-cloud-controller-manager +spec: + selector: + matchLabels: + k8s-app: openstack-cloud-controller-manager + updateStrategy: + type: RollingUpdate + template: + metadata: + labels: + k8s-app: openstack-cloud-controller-manager + spec: + nodeSelector: + node-role.kubernetes.io/control-plane: "" + securityContext: + runAsUser: 999 + tolerations: + - key: node.cloudprovider.kubernetes.io/uninitialized + value: "true" + effect: NoSchedule + - key: node-role.kubernetes.io/master + effect: NoSchedule + - key: node-role.kubernetes.io/control-plane + effect: NoSchedule + serviceAccountName: cloud-controller-manager + containers: + - name: openstack-cloud-controller-manager + image: {{ docker_image_repo }}/k8scloudprovider/openstack-cloud-controller-manager:{{ external_openstack_cloud_controller_image_tag }} + args: + - /bin/openstack-cloud-controller-manager + - --v=1 + - --cloud-config=$(CLOUD_CONFIG) + - --cloud-provider=openstack + - --cluster-name={{ cluster_name }} + - --use-service-account-credentials=true + - --bind-address={{ external_openstack_cloud_controller_bind_address }} +{% for key, value in external_openstack_cloud_controller_extra_args.items() %} + - "{{ '--' + key + '=' + value }}" +{% endfor %} + volumeMounts: + - mountPath: /etc/kubernetes/pki + name: k8s-certs + readOnly: true + - mountPath: /etc/ssl/certs + name: ca-certs + readOnly: true +{% if ssl_ca_dirs | length %} +{% for dir in ssl_ca_dirs %} + - name: {{ dir | regex_replace('^/(.*)$', '\\1' ) | regex_replace('/', '-') }} + mountPath: {{ dir }} + readOnly: true +{% endfor %} +{% endif %} + - mountPath: /etc/config/cloud.conf + name: cloud-config-volume + readOnly: true + subPath: cloud.conf + - mountPath: {{ kube_config_dir }}/external-openstack-cacert.pem + name: cloud-config-volume + readOnly: true + subPath: ca.cert +{% if kubelet_flexvolumes_plugins_dir is defined %} + - mountPath: /usr/libexec/kubernetes/kubelet-plugins/volume/exec + name: flexvolume-dir +{% endif %} + resources: + requests: + cpu: 200m + env: + - name: CLOUD_CONFIG + value: /etc/config/cloud.conf + hostNetwork: true + volumes: +{% if kubelet_flexvolumes_plugins_dir is defined %} + - name: flexvolume-dir + hostPath: + path: "{{ kubelet_flexvolumes_plugins_dir }}" + type: DirectoryOrCreate +{% endif %} + - name: k8s-certs + hostPath: + path: /etc/kubernetes/pki + type: DirectoryOrCreate + - name: ca-certs + hostPath: + path: /etc/ssl/certs + type: DirectoryOrCreate +{% if ssl_ca_dirs | length %} +{% for dir in ssl_ca_dirs %} + - name: {{ dir | regex_replace('^/(.*)$', '\\1' ) | regex_replace('/', '-') }} + hostPath: + path: {{ dir }} + type: DirectoryOrCreate +{% endfor %} +{% endif %} + - name: cloud-config-volume + secret: + secretName: external-openstack-cloud-config diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-controller-manager-role-bindings.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-controller-manager-role-bindings.yml.j2 new file mode 100644 index 0000000..bbdf336 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-controller-manager-role-bindings.yml.j2 @@ -0,0 +1,16 @@ +apiVersion: v1 +items: +- apiVersion: rbac.authorization.k8s.io/v1 + kind: ClusterRoleBinding + metadata: + name: system:cloud-controller-manager + roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: system:cloud-controller-manager + subjects: + - kind: ServiceAccount + name: cloud-controller-manager + namespace: kube-system +kind: List +metadata: {} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-controller-manager-roles.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-controller-manager-roles.yml.j2 new file mode 100644 index 0000000..2ab3a5b --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/openstack/templates/external-openstack-cloud-controller-manager-roles.yml.j2 @@ -0,0 +1,109 @@ +apiVersion: v1 +items: +- apiVersion: rbac.authorization.k8s.io/v1 + kind: ClusterRole + metadata: + name: system:cloud-controller-manager + rules: + - apiGroups: + - coordination.k8s.io + resources: + - leases + verbs: + - get + - create + - update + - apiGroups: + - "" + resources: + - events + verbs: + - create + - patch + - update + - apiGroups: + - "" + resources: + - nodes + verbs: + - '*' + - apiGroups: + - "" + resources: + - nodes/status + verbs: + - patch + - apiGroups: + - "" + resources: + - services + verbs: + - list + - patch + - update + - watch + - apiGroups: + - "" + resources: + - services/status + verbs: + - patch + - apiGroups: + - "" + resources: + - serviceaccounts/token + verbs: + - create + - apiGroups: + - "" + resources: + - serviceaccounts + verbs: + - create + - get + - apiGroups: + - "" + resources: + - persistentvolumes + verbs: + - '*' + - apiGroups: + - "" + resources: + - endpoints + verbs: + - create + - get + - list + - watch + - update + - apiGroups: + - "" + resources: + - configmaps + verbs: + - get + - list + - watch + - apiGroups: + - "" + resources: + - secrets + verbs: + - list + - get + - watch + - apiGroups: + - authentication.k8s.io + resources: + - tokenreviews + verbs: + - create + - apiGroups: + - authorization.k8s.io + resources: + - subjectaccessreviews + verbs: + - create +kind: List +metadata: {} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/defaults/main.yml b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/defaults/main.yml new file mode 100644 index 0000000..b6fb797 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/defaults/main.yml @@ -0,0 +1,14 @@ +--- +external_vsphere_vcenter_port: "443" +external_vsphere_insecure: "true" + +## A dictionary of extra arguments to add to the vsphere cloud controller manager daemonset +## Format: +## external_vsphere_cloud_controller_extra_args: +## arg1: "value1" +## arg2: "value2" +external_vsphere_cloud_controller_extra_args: {} +external_vsphere_cloud_controller_image_tag: "latest" + +external_vsphere_user: "{{ lookup('env', 'VSPHERE_USER') }}" +external_vsphere_password: "{{ lookup('env', 'VSPHERE_PASSWORD') }}" diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/tasks/main.yml b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/tasks/main.yml new file mode 100644 index 0000000..60b8ec8 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/tasks/main.yml @@ -0,0 +1,49 @@ +--- +- name: External vSphere Cloud Controller | Check vsphere credentials + include_tasks: vsphere-credentials-check.yml + +- name: External vSphere Cloud Controller | Generate CPI cloud-config + template: + src: "{{ item }}.j2" + dest: "{{ kube_config_dir }}/{{ item }}" + mode: 0640 + with_items: + - external-vsphere-cpi-cloud-config + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: External vSphere Cloud Controller | Generate Manifests + template: + src: "{{ item }}.j2" + dest: "{{ kube_config_dir }}/{{ item }}" + mode: 0644 + with_items: + - external-vsphere-cpi-cloud-config-secret.yml + - external-vsphere-cloud-controller-manager-roles.yml + - external-vsphere-cloud-controller-manager-role-bindings.yml + - external-vsphere-cloud-controller-manager-ds.yml + register: external_vsphere_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: External vSphere Cloud Provider Interface | Create a CPI configMap manifest + command: "{{ bin_dir }}/kubectl create configmap cloud-config --from-file=vsphere.conf={{ kube_config_dir }}/external-vsphere-cpi-cloud-config -n kube-system --dry-run --save-config -o yaml" + register: external_vsphere_configmap_manifest + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: External vSphere Cloud Provider Interface | Apply a CPI configMap manifest + command: + cmd: "{{ bin_dir }}/kubectl apply -f -" + stdin: "{{ external_vsphere_configmap_manifest.stdout }}" + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: External vSphere Cloud Controller | Apply Manifests + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.item }}" + state: "latest" + with_items: + - "{{ external_vsphere_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item }}" diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/tasks/vsphere-credentials-check.yml b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/tasks/vsphere-credentials-check.yml new file mode 100644 index 0000000..b6c12b8 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/tasks/vsphere-credentials-check.yml @@ -0,0 +1,32 @@ +--- +- name: External vSphere Cloud Provider | check external_vsphere_vcenter_ip value + fail: + msg: "external_vsphere_vcenter_ip is missing" + when: external_vsphere_vcenter_ip is not defined or not external_vsphere_vcenter_ip + +- name: External vSphere Cloud Provider | check external_vsphere_vcenter_port value + fail: + msg: "external_vsphere_vcenter_port is missing" + when: external_vsphere_vcenter_port is not defined or not external_vsphere_vcenter_port + +- name: External vSphere Cloud Provider | check external_vsphere_insecure value + fail: + msg: "external_vsphere_insecure is missing" + when: external_vsphere_insecure is not defined or not external_vsphere_insecure + +- name: External vSphere Cloud Provider | check external_vsphere_user value + fail: + msg: "external_vsphere_user is missing" + when: external_vsphere_user is not defined or not external_vsphere_user + +- name: External vSphere Cloud Provider | check external_vsphere_password value + fail: + msg: "external_vsphere_password is missing" + when: + - external_vsphere_password is not defined or not external_vsphere_password + +- name: External vSphere Cloud Provider | check external_vsphere_datacenter value + fail: + msg: "external_vsphere_datacenter is missing" + when: + - external_vsphere_datacenter is not defined or not external_vsphere_datacenter diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cloud-controller-manager-ds.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cloud-controller-manager-ds.yml.j2 new file mode 100644 index 0000000..5f1068d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cloud-controller-manager-ds.yml.j2 @@ -0,0 +1,76 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: cloud-controller-manager + namespace: kube-system +--- +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: vsphere-cloud-controller-manager + namespace: kube-system + labels: + k8s-app: vsphere-cloud-controller-manager +spec: + selector: + matchLabels: + k8s-app: vsphere-cloud-controller-manager + updateStrategy: + type: RollingUpdate + template: + metadata: + labels: + k8s-app: vsphere-cloud-controller-manager + spec: + nodeSelector: + node-role.kubernetes.io/control-plane: "" + securityContext: + runAsUser: 0 + tolerations: + - key: node.cloudprovider.kubernetes.io/uninitialized + value: "true" + effect: NoSchedule + - key: node-role.kubernetes.io/master + effect: NoSchedule + - key: node-role.kubernetes.io/control-plane + effect: NoSchedule + serviceAccountName: cloud-controller-manager + containers: + - name: vsphere-cloud-controller-manager + image: {{ gcr_image_repo }}/cloud-provider-vsphere/cpi/release/manager:{{ external_vsphere_cloud_controller_image_tag }} + args: + - --v=2 + - --cloud-provider=vsphere + - --cloud-config=/etc/cloud/vsphere.conf +{% for key, value in external_vsphere_cloud_controller_extra_args.items() %} + - "{{ '--' + key + '=' + value }}" +{% endfor %} + volumeMounts: + - mountPath: /etc/cloud + name: vsphere-config-volume + readOnly: true + resources: + requests: + cpu: 200m + hostNetwork: true + volumes: + - name: vsphere-config-volume + configMap: + name: cloud-config +--- +apiVersion: v1 +kind: Service +metadata: + labels: + component: cloud-controller-manager + name: vsphere-cloud-controller-manager + namespace: kube-system +spec: + type: NodePort + ports: + - port: 43001 + protocol: TCP + targetPort: 43001 + selector: + component: cloud-controller-manager diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cloud-controller-manager-role-bindings.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cloud-controller-manager-role-bindings.yml.j2 new file mode 100644 index 0000000..9f6107d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cloud-controller-manager-role-bindings.yml.j2 @@ -0,0 +1,35 @@ +apiVersion: v1 +items: +- apiVersion: rbac.authorization.k8s.io/v1 + kind: RoleBinding + metadata: + name: servicecatalog.k8s.io:apiserver-authentication-reader + namespace: kube-system + roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: extension-apiserver-authentication-reader + subjects: + - apiGroup: "" + kind: ServiceAccount + name: cloud-controller-manager + namespace: kube-system + - apiGroup: "" + kind: User + name: cloud-controller-manager +- apiVersion: rbac.authorization.k8s.io/v1 + kind: ClusterRoleBinding + metadata: + name: system:cloud-controller-manager + roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: system:cloud-controller-manager + subjects: + - kind: ServiceAccount + name: cloud-controller-manager + namespace: kube-system + - kind: User + name: cloud-controller-manager +kind: List +metadata: {} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cloud-controller-manager-roles.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cloud-controller-manager-roles.yml.j2 new file mode 100644 index 0000000..2cd7ad0 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cloud-controller-manager-roles.yml.j2 @@ -0,0 +1,91 @@ +apiVersion: v1 +items: +- apiVersion: rbac.authorization.k8s.io/v1 + kind: ClusterRole + metadata: + name: system:cloud-controller-manager + rules: + - apiGroups: + - "" + resources: + - events + verbs: + - create + - patch + - update + - apiGroups: + - "" + resources: + - nodes + verbs: + - '*' + - apiGroups: + - "" + resources: + - nodes/status + verbs: + - patch + - apiGroups: + - "" + resources: + - services + verbs: + - list + - patch + - update + - watch + - apiGroups: + - "" + resources: + - services/status + verbs: + - patch + - apiGroups: + - "" + resources: + - serviceaccounts + verbs: + - create + - get + - list + - watch + - update + - apiGroups: + - "" + resources: + - persistentvolumes + verbs: + - get + - list + - update + - watch + - apiGroups: + - "" + resources: + - endpoints + verbs: + - create + - get + - list + - watch + - update + - apiGroups: + - "" + resources: + - secrets + verbs: + - get + - list + - watch + - apiGroups: + - "coordination.k8s.io" + resources: + - leases + verbs: + - get + - list + - watch + - create + - update +kind: List +metadata: {} diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cpi-cloud-config-secret.yml.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cpi-cloud-config-secret.yml.j2 new file mode 100644 index 0000000..5364f42 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cpi-cloud-config-secret.yml.j2 @@ -0,0 +1,11 @@ +# This YAML file contains secret objects, +# which are necessary to run external vsphere cloud controller. + +apiVersion: v1 +kind: Secret +metadata: + name: cpi-global-secret + namespace: kube-system +stringData: + {{ external_vsphere_vcenter_ip }}.username: "{{ external_vsphere_user }}" + {{ external_vsphere_vcenter_ip }}.password: "{{ external_vsphere_password }}" diff --git a/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cpi-cloud-config.j2 b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cpi-cloud-config.j2 new file mode 100644 index 0000000..a32d876 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_cloud_controller/vsphere/templates/external-vsphere-cpi-cloud-config.j2 @@ -0,0 +1,8 @@ +[Global] +port = "{{ external_vsphere_vcenter_port }}" +insecure-flag = "{{ external_vsphere_insecure }}" +secret-name = "cpi-global-secret" +secret-namespace = "kube-system" + +[VirtualCenter "{{ external_vsphere_vcenter_ip }}"] +datacenters = "{{ external_vsphere_datacenter }}" diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/defaults/main.yml b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/defaults/main.yml new file mode 100644 index 0000000..577fbff --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/defaults/main.yml @@ -0,0 +1,10 @@ +--- +cephfs_provisioner_namespace: "cephfs-provisioner" +cephfs_provisioner_cluster: ceph +cephfs_provisioner_monitors: ~ +cephfs_provisioner_admin_id: admin +cephfs_provisioner_secret: secret +cephfs_provisioner_storage_class: cephfs +cephfs_provisioner_reclaim_policy: Delete +cephfs_provisioner_claim_root: /volumes +cephfs_provisioner_deterministic_names: true diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/tasks/main.yml b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/tasks/main.yml new file mode 100644 index 0000000..95a2f75 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/tasks/main.yml @@ -0,0 +1,80 @@ +--- + +- name: CephFS Provisioner | Remove legacy addon dir and manifests + file: + path: "{{ kube_config_dir }}/addons/cephfs_provisioner" + state: absent + when: + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - upgrade + +- name: CephFS Provisioner | Remove legacy namespace + command: > + {{ kubectl }} delete namespace {{ cephfs_provisioner_namespace }} + ignore_errors: true # noqa ignore-errors + when: + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - upgrade + +- name: CephFS Provisioner | Remove legacy storageclass + command: > + {{ kubectl }} delete storageclass {{ cephfs_provisioner_storage_class }} + ignore_errors: true # noqa ignore-errors + when: + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - upgrade + +- name: CephFS Provisioner | Create addon dir + file: + path: "{{ kube_config_dir }}/addons/cephfs_provisioner" + state: directory + owner: root + group: root + mode: 0755 + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: CephFS Provisioner | Templates list + set_fact: + cephfs_provisioner_templates: + - { name: 00-namespace, file: 00-namespace.yml, type: ns } + - { name: secret-cephfs-provisioner, file: secret-cephfs-provisioner.yml, type: secret } + - { name: sa-cephfs-provisioner, file: sa-cephfs-provisioner.yml, type: sa } + - { name: clusterrole-cephfs-provisioner, file: clusterrole-cephfs-provisioner.yml, type: clusterrole } + - { name: clusterrolebinding-cephfs-provisioner, file: clusterrolebinding-cephfs-provisioner.yml, type: clusterrolebinding } + - { name: role-cephfs-provisioner, file: role-cephfs-provisioner.yml, type: role } + - { name: rolebinding-cephfs-provisioner, file: rolebinding-cephfs-provisioner.yml, type: rolebinding } + - { name: deploy-cephfs-provisioner, file: deploy-cephfs-provisioner.yml, type: deploy } + - { name: sc-cephfs-provisioner, file: sc-cephfs-provisioner.yml, type: sc } + cephfs_provisioner_templates_for_psp: + - { name: psp-cephfs-provisioner, file: psp-cephfs-provisioner.yml, type: psp } + +- name: CephFS Provisioner | Append extra templates to CephFS Provisioner Templates list for PodSecurityPolicy + set_fact: + cephfs_provisioner_templates: "{{ cephfs_provisioner_templates_for_psp + cephfs_provisioner_templates }}" + when: + - podsecuritypolicy_enabled + - cephfs_provisioner_namespace != "kube-system" + +- name: CephFS Provisioner | Create manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/cephfs_provisioner/{{ item.file }}" + mode: 0644 + with_items: "{{ cephfs_provisioner_templates }}" + register: cephfs_provisioner_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: CephFS Provisioner | Apply manifests + kube: + name: "{{ item.item.name }}" + namespace: "{{ cephfs_provisioner_namespace }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/cephfs_provisioner/{{ item.item.file }}" + state: "latest" + with_items: "{{ cephfs_provisioner_manifests.results }}" + when: inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/00-namespace.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/00-namespace.yml.j2 new file mode 100644 index 0000000..2a2a67c --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/00-namespace.yml.j2 @@ -0,0 +1,7 @@ +--- +apiVersion: v1 +kind: Namespace +metadata: + name: {{ cephfs_provisioner_namespace }} + labels: + name: {{ cephfs_provisioner_namespace }} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/clusterrole-cephfs-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/clusterrole-cephfs-provisioner.yml.j2 new file mode 100644 index 0000000..4c92ea6 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/clusterrole-cephfs-provisioner.yml.j2 @@ -0,0 +1,26 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cephfs-provisioner + namespace: {{ cephfs_provisioner_namespace }} +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "create", "delete"] + - apiGroups: [""] + resources: ["persistentvolumeclaims"] + verbs: ["get", "list", "watch", "update"] + - apiGroups: ["storage.k8s.io"] + resources: ["storageclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["create", "update", "patch"] + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "create", "delete"] + - apiGroups: ["policy"] + resourceNames: ["cephfs-provisioner"] + resources: ["podsecuritypolicies"] + verbs: ["use"] diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/clusterrolebinding-cephfs-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/clusterrolebinding-cephfs-provisioner.yml.j2 new file mode 100644 index 0000000..cc5d5ff --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/clusterrolebinding-cephfs-provisioner.yml.j2 @@ -0,0 +1,13 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cephfs-provisioner +subjects: + - kind: ServiceAccount + name: cephfs-provisioner + namespace: {{ cephfs_provisioner_namespace }} +roleRef: + kind: ClusterRole + name: cephfs-provisioner + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/deploy-cephfs-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/deploy-cephfs-provisioner.yml.j2 new file mode 100644 index 0000000..8d9eb08 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/deploy-cephfs-provisioner.yml.j2 @@ -0,0 +1,34 @@ +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + name: cephfs-provisioner + namespace: {{ cephfs_provisioner_namespace }} + labels: + app: cephfs-provisioner + version: {{ cephfs_provisioner_image_tag }} +spec: + replicas: 1 + selector: + matchLabels: + app: cephfs-provisioner + version: {{ cephfs_provisioner_image_tag }} + template: + metadata: + labels: + app: cephfs-provisioner + version: {{ cephfs_provisioner_image_tag }} + spec: + priorityClassName: {% if cephfs_provisioner_namespace == 'kube-system' %}system-cluster-critical{% else %}k8s-cluster-critical{% endif %}{{ '' }} + serviceAccount: cephfs-provisioner + containers: + - name: cephfs-provisioner + image: {{ cephfs_provisioner_image_repo }}:{{ cephfs_provisioner_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + env: + - name: PROVISIONER_NAME + value: ceph.com/cephfs + command: + - "/usr/local/bin/cephfs-provisioner" + args: + - "-id=cephfs-provisioner-1" diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/psp-cephfs-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/psp-cephfs-provisioner.yml.j2 new file mode 100644 index 0000000..76d146c --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/psp-cephfs-provisioner.yml.j2 @@ -0,0 +1,44 @@ +--- +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: cephfs-provisioner + annotations: + seccomp.security.alpha.kubernetes.io/defaultProfileName: 'runtime/default' + seccomp.security.alpha.kubernetes.io/allowedProfileNames: 'runtime/default' +{% if apparmor_enabled %} + apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + apparmor.security.beta.kubernetes.io/allowedProfileNames: 'runtime/default' +{% endif %} + labels: + addonmanager.kubernetes.io/mode: Reconcile +spec: + privileged: false + allowPrivilegeEscalation: false + requiredDropCapabilities: + - ALL + volumes: + - 'configMap' + - 'emptyDir' + - 'projected' + - 'secret' + - 'downwardAPI' + - 'persistentVolumeClaim' + hostNetwork: false + hostIPC: false + hostPID: false + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + readOnlyRootFilesystem: false diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/role-cephfs-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/role-cephfs-provisioner.yml.j2 new file mode 100644 index 0000000..1fb80a1 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/role-cephfs-provisioner.yml.j2 @@ -0,0 +1,13 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: cephfs-provisioner + namespace: {{ cephfs_provisioner_namespace }} +rules: + - apiGroups: [""] + resources: ["secrets"] + verbs: ["create", "get", "delete"] + - apiGroups: [""] + resources: ["endpoints"] + verbs: ["get", "list", "watch", "create", "update", "patch"] diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/rolebinding-cephfs-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/rolebinding-cephfs-provisioner.yml.j2 new file mode 100644 index 0000000..01ab87b --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/rolebinding-cephfs-provisioner.yml.j2 @@ -0,0 +1,14 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: cephfs-provisioner + namespace: {{ cephfs_provisioner_namespace }} +subjects: + - kind: ServiceAccount + name: cephfs-provisioner + namespace: {{ cephfs_provisioner_namespace }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: cephfs-provisioner diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/sa-cephfs-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/sa-cephfs-provisioner.yml.j2 new file mode 100644 index 0000000..31f87bd --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/sa-cephfs-provisioner.yml.j2 @@ -0,0 +1,6 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: cephfs-provisioner + namespace: {{ cephfs_provisioner_namespace }} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/sc-cephfs-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/sc-cephfs-provisioner.yml.j2 new file mode 100644 index 0000000..dd0e37e --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/sc-cephfs-provisioner.yml.j2 @@ -0,0 +1,15 @@ +--- +apiVersion: storage.k8s.io/v1 +kind: StorageClass +metadata: + name: {{ cephfs_provisioner_storage_class }} +provisioner: ceph.com/cephfs +reclaimPolicy: {{ cephfs_provisioner_reclaim_policy }} +parameters: + cluster: {{ cephfs_provisioner_cluster }} + monitors: {{ cephfs_provisioner_monitors }} + adminId: {{ cephfs_provisioner_admin_id }} + adminSecretName: cephfs-provisioner + adminSecretNamespace: {{ cephfs_provisioner_namespace }} + claimRoot: {{ cephfs_provisioner_claim_root }} + deterministicNames: "{{ cephfs_provisioner_deterministic_names | bool | lower }}" diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/secret-cephfs-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/secret-cephfs-provisioner.yml.j2 new file mode 100644 index 0000000..6d73c0c --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/cephfs_provisioner/templates/secret-cephfs-provisioner.yml.j2 @@ -0,0 +1,9 @@ +--- +kind: Secret +apiVersion: v1 +metadata: + name: cephfs-provisioner + namespace: {{ cephfs_provisioner_namespace }} +type: Opaque +data: + secret: {{ cephfs_provisioner_secret | b64encode }} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/defaults/main.yml b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/defaults/main.yml new file mode 100644 index 0000000..aacfb22 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/defaults/main.yml @@ -0,0 +1,10 @@ +--- +local_path_provisioner_enabled: false +local_path_provisioner_namespace: "local-path-storage" +local_path_provisioner_storage_class: "local-path" +local_path_provisioner_reclaim_policy: Delete +local_path_provisioner_claim_root: /opt/local-path-provisioner/ +local_path_provisioner_is_default_storageclass: "true" +local_path_provisioner_debug: false +local_path_provisioner_helper_image_repo: "busybox" +local_path_provisioner_helper_image_tag: "latest" diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/tasks/main.yml b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/tasks/main.yml new file mode 100644 index 0000000..71036ca --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/tasks/main.yml @@ -0,0 +1,47 @@ +--- +- name: Local Path Provisioner | Create addon dir + file: + path: "{{ kube_config_dir }}/addons/local_path_provisioner" + state: directory + owner: root + group: root + mode: 0755 + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Local Path Provisioner | Create claim root dir + file: + path: "{{ local_path_provisioner_claim_root }}" + state: directory + mode: 0755 + +- name: Local Path Provisioner | Render Template + set_fact: + local_path_provisioner_templates: + - { name: local-path-storage-ns, file: local-path-storage-ns.yml, type: ns } + - { name: local-path-storage-sa, file: local-path-storage-sa.yml, type: sa } + - { name: local-path-storage-cr, file: local-path-storage-cr.yml, type: cr } + - { name: local-path-storage-clusterrolebinding, file: local-path-storage-clusterrolebinding.yml, type: clusterrolebinding } + - { name: local-path-storage-cm, file: local-path-storage-cm.yml, type: cm } + - { name: local-path-storage-deployment, file: local-path-storage-deployment.yml, type: deployment } + - { name: local-path-storage-sc, file: local-path-storage-sc.yml, type: sc } + +- name: Local Path Provisioner | Create manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/local_path_provisioner/{{ item.file }}" + mode: 0644 + with_items: "{{ local_path_provisioner_templates }}" + register: local_path_provisioner_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Local Path Provisioner | Apply manifests + kube: + name: "{{ item.item.name }}" + namespace: "{{ local_path_provisioner_namespace }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/local_path_provisioner/{{ item.item.file }}" + state: "latest" + with_items: "{{ local_path_provisioner_manifests.results }}" + when: inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-clusterrolebinding.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-clusterrolebinding.yml.j2 new file mode 100644 index 0000000..317a71f --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-clusterrolebinding.yml.j2 @@ -0,0 +1,13 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: local-path-provisioner-bind +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: local-path-provisioner-role +subjects: + - kind: ServiceAccount + name: local-path-provisioner-service-account + namespace: {{ local_path_provisioner_namespace }} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-cm.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-cm.yml.j2 new file mode 100644 index 0000000..9cd7fd3 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-cm.yml.j2 @@ -0,0 +1,35 @@ +--- +kind: ConfigMap +apiVersion: v1 +metadata: + name: local-path-config + namespace: {{ local_path_provisioner_namespace }} +data: + config.json: |- + { + "nodePathMap":[ + { + "node":"DEFAULT_PATH_FOR_NON_LISTED_NODES", + "paths":["{{ local_path_provisioner_claim_root }}"] + } + ] + } + setup: |- + #!/bin/sh + set -eu + mkdir -m 0777 -p "$VOL_DIR" + teardown: |- + #!/bin/sh + set -eu + rm -rf "$VOL_DIR" + helperPod.yaml: |- + apiVersion: v1 + kind: Pod + metadata: + name: helper-pod + spec: + containers: + - name: helper-pod + image: "{{ local_path_provisioner_helper_image_repo }}:{{ local_path_provisioner_helper_image_tag }}" + imagePullPolicy: IfNotPresent + diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-cr.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-cr.yml.j2 new file mode 100644 index 0000000..299db6e --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-cr.yml.j2 @@ -0,0 +1,18 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: local-path-provisioner-role +rules: + - apiGroups: [ "" ] + resources: [ "nodes", "persistentvolumeclaims", "configmaps" ] + verbs: [ "get", "list", "watch" ] + - apiGroups: [ "" ] + resources: [ "endpoints", "persistentvolumes", "pods" ] + verbs: [ "*" ] + - apiGroups: [ "" ] + resources: [ "events" ] + verbs: [ "create", "patch" ] + - apiGroups: [ "storage.k8s.io" ] + resources: [ "storageclasses" ] + verbs: [ "get", "list", "watch" ] \ No newline at end of file diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-deployment.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-deployment.yml.j2 new file mode 100644 index 0000000..6ce426a --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-deployment.yml.j2 @@ -0,0 +1,41 @@ +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + name: local-path-provisioner + namespace: {{ local_path_provisioner_namespace }} +spec: + replicas: 1 + selector: + matchLabels: + app: local-path-provisioner + template: + metadata: + labels: + app: local-path-provisioner + spec: + serviceAccountName: local-path-provisioner-service-account + containers: + - name: local-path-provisioner + image: {{ local_path_provisioner_image_repo }}:{{ local_path_provisioner_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + command: + - local-path-provisioner + - start + - --config + - /etc/config/config.json +{% if local_path_provisioner_debug | default(false) %} + - --debug +{% endif %} + volumeMounts: + - name: config-volume + mountPath: /etc/config/ + env: + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + volumes: + - name: config-volume + configMap: + name: local-path-config diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-ns.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-ns.yml.j2 new file mode 100644 index 0000000..1e8c6ce --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-ns.yml.j2 @@ -0,0 +1,5 @@ +--- +apiVersion: v1 +kind: Namespace +metadata: + name: {{ local_path_provisioner_namespace }} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-sa.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-sa.yml.j2 new file mode 100644 index 0000000..128a106 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-sa.yml.j2 @@ -0,0 +1,6 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: local-path-provisioner-service-account + namespace: {{ local_path_provisioner_namespace }} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-sc.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-sc.yml.j2 new file mode 100644 index 0000000..d662661 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_path_provisioner/templates/local-path-storage-sc.yml.j2 @@ -0,0 +1,10 @@ +--- +apiVersion: storage.k8s.io/v1 +kind: StorageClass +metadata: + name: {{ local_path_provisioner_storage_class }} + annotations: + storageclass.kubernetes.io/is-default-class: "{{ local_path_provisioner_is_default_storageclass }}" +provisioner: rancher.io/local-path +volumeBindingMode: WaitForFirstConsumer +reclaimPolicy: {{ local_path_provisioner_reclaim_policy }} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/defaults/main.yml b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/defaults/main.yml new file mode 100644 index 0000000..38afefb --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/defaults/main.yml @@ -0,0 +1,20 @@ +--- +local_volume_provisioner_namespace: "kube-system" +# List of node labels to be copied to the PVs created by the provisioner +local_volume_provisioner_nodelabels: [] +# - kubernetes.io/hostname +# - topology.kubernetes.io/region +# - topology.kubernetes.io/zone +local_volume_provisioner_tolerations: [] +local_volume_provisioner_use_node_name_only: false +# Leverages Ansible's string to Python datatype casting. Otherwise the dict_key isn't substituted. +# see https://github.com/ansible/ansible/issues/17324 +local_volume_provisioner_storage_classes: | + { + "{{ local_volume_provisioner_storage_class | default('local-storage') }}": { + "host_dir": "{{ local_volume_provisioner_base_dir | default('/mnt/disks') }}", + "mount_dir": "{{ local_volume_provisioner_mount_dir | default('/mnt/disks') }}", + "volume_mode": "Filesystem", + "fs_type": "ext4" + } + } diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/tasks/basedirs.yml b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/tasks/basedirs.yml new file mode 100644 index 0000000..7add2da --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/tasks/basedirs.yml @@ -0,0 +1,12 @@ +--- +# include to workaround mitogen issue +# https://github.com/dw/mitogen/issues/663 + +- name: "Local Volume Provisioner | Ensure base dir {{ delegate_host_base_dir.1 }} is created on {{ delegate_host_base_dir.0 }}" + file: + path: "{{ local_volume_provisioner_storage_classes[delegate_host_base_dir.1].host_dir }}" + state: directory + owner: root + group: root + mode: "{{ local_volume_provisioner_directory_mode }}" + delegate_to: "{{ delegate_host_base_dir.0 }}" diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/tasks/main.yml b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/tasks/main.yml new file mode 100644 index 0000000..2308b5c --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/tasks/main.yml @@ -0,0 +1,48 @@ +--- + +- name: Local Volume Provisioner | Ensure base dir is created on all hosts + include_tasks: basedirs.yml + loop_control: + loop_var: delegate_host_base_dir + loop: "{{ groups['k8s_cluster'] | product(local_volume_provisioner_storage_classes.keys()) | list }}" + +- name: Local Volume Provisioner | Create addon dir + file: + path: "{{ kube_config_dir }}/addons/local_volume_provisioner" + state: directory + owner: root + group: root + mode: 0755 + +- name: Local Volume Provisioner | Templates list + set_fact: + local_volume_provisioner_templates: + - { name: local-volume-provisioner-ns, file: local-volume-provisioner-ns.yml, type: ns } + - { name: local-volume-provisioner-sa, file: local-volume-provisioner-sa.yml, type: sa } + - { name: local-volume-provisioner-clusterrole, file: local-volume-provisioner-clusterrole.yml, type: clusterrole } + - { name: local-volume-provisioner-clusterrolebinding, file: local-volume-provisioner-clusterrolebinding.yml, type: clusterrolebinding } + - { name: local-volume-provisioner-cm, file: local-volume-provisioner-cm.yml, type: cm } + - { name: local-volume-provisioner-ds, file: local-volume-provisioner-ds.yml, type: ds } + - { name: local-volume-provisioner-sc, file: local-volume-provisioner-sc.yml, type: sc } + +- name: Local Volume Provisioner | Create manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/local_volume_provisioner/{{ item.file }}" + mode: 0644 + with_items: "{{ local_volume_provisioner_templates }}" + register: local_volume_provisioner_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Local Volume Provisioner | Apply manifests + kube: + name: "{{ item.item.name }}" + namespace: "{{ local_volume_provisioner_namespace }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/local_volume_provisioner/{{ item.item.file }}" + state: "latest" + with_items: "{{ local_volume_provisioner_manifests.results }}" + when: inventory_hostname == groups['kube_control_plane'][0] + loop_control: + label: "{{ item.item.file }}" diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-clusterrole.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-clusterrole.yml.j2 new file mode 100644 index 0000000..ada55dd --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-clusterrole.yml.j2 @@ -0,0 +1,22 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: local-volume-provisioner-node-clusterrole + namespace: {{ local_volume_provisioner_namespace }} +rules: +- apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "create", "delete"] +- apiGroups: ["storage.k8s.io"] + resources: ["storageclasses"] + verbs: ["get", "list", "watch"] +- apiGroups: [""] + resources: ["events"] + verbs: ["watch"] +- apiGroups: ["", "events.k8s.io"] + resources: ["events"] + verbs: ["create", "update", "patch"] +- apiGroups: [""] + resources: ["nodes"] + verbs: ["get"] diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-clusterrolebinding.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-clusterrolebinding.yml.j2 new file mode 100644 index 0000000..bc286b2 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-clusterrolebinding.yml.j2 @@ -0,0 +1,14 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: local-volume-provisioner-system-node + namespace: {{ local_volume_provisioner_namespace }} +subjects: +- kind: ServiceAccount + name: local-volume-provisioner + namespace: {{ local_volume_provisioner_namespace }} +roleRef: + kind: ClusterRole + name: local-volume-provisioner-node-clusterrole + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-cm.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-cm.yml.j2 new file mode 100644 index 0000000..7e37283 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-cm.yml.j2 @@ -0,0 +1,33 @@ +# Macro to convert camelCase dictionary keys to snake_case keys +{% macro convert_keys(mydict) -%} + {% for key in mydict.keys() | list -%} + {% set key_split = key.split('_') -%} + {% set new_key = key_split[0] + key_split[1:] | map('capitalize') | join -%} + {% set value = mydict.pop(key) -%} + {{ mydict.__setitem__(new_key, value) -}} + {{ convert_keys(value) if value is mapping else None -}} + {% endfor -%} +{% endmacro -%} + +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: local-volume-provisioner + namespace: {{ local_volume_provisioner_namespace }} +data: +{% if local_volume_provisioner_nodelabels | length > 0 %} + nodeLabelsForPV: | +{% for nodelabel in local_volume_provisioner_nodelabels %} + - {{ nodelabel }} +{% endfor %} +{% endif %} +{% if local_volume_provisioner_use_node_name_only %} + useNodeNameOnly: "true" +{% endif %} + storageClassMap: | +{% for class_name, storage_class in local_volume_provisioner_storage_classes.items() %} + {{ class_name }}: + {{- convert_keys(storage_class) }} + {{ storage_class | to_nice_yaml(indent=2) | indent(6) }} +{%- endfor %} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-ds.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-ds.yml.j2 new file mode 100644 index 0000000..90a4730 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-ds.yml.j2 @@ -0,0 +1,66 @@ +--- +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: local-volume-provisioner + namespace: {{ local_volume_provisioner_namespace }} + labels: + k8s-app: local-volume-provisioner + version: {{ local_volume_provisioner_image_tag }} +spec: + selector: + matchLabels: + k8s-app: local-volume-provisioner + version: {{ local_volume_provisioner_image_tag }} + template: + metadata: + labels: + k8s-app: local-volume-provisioner + version: {{ local_volume_provisioner_image_tag }} + spec: + priorityClassName: {% if local_volume_provisioner_namespace == 'kube-system' %}system-node-critical{% else %}k8s-cluster-critical{% endif %}{{ '' }} + serviceAccountName: local-volume-provisioner + nodeSelector: + kubernetes.io/os: linux +{% if local_volume_provisioner_tolerations %} + tolerations: + {{ local_volume_provisioner_tolerations | to_nice_yaml(indent=2) | indent(width=8) }} +{% endif %} + containers: + - name: provisioner + image: {{ local_volume_provisioner_image_repo }}:{{ local_volume_provisioner_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + securityContext: + privileged: true + env: + - name: MY_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: MY_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + volumeMounts: + - name: local-volume-provisioner + mountPath: /etc/provisioner/config + readOnly: true + - mountPath: /dev + name: provisioner-dev +{% for class_name, class_config in local_volume_provisioner_storage_classes.items() %} + - name: local-volume-provisioner-hostpath-{{ class_name }} + mountPath: {{ class_config.mount_dir }} + mountPropagation: "HostToContainer" +{% endfor %} + volumes: + - name: local-volume-provisioner + configMap: + name: local-volume-provisioner + - name: provisioner-dev + hostPath: + path: /dev +{% for class_name, class_config in local_volume_provisioner_storage_classes.items() %} + - name: local-volume-provisioner-hostpath-{{ class_name }} + hostPath: + path: {{ class_config.host_dir }} +{% endfor %} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-ns.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-ns.yml.j2 new file mode 100644 index 0000000..04a7910 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-ns.yml.j2 @@ -0,0 +1,7 @@ +--- +apiVersion: v1 +kind: Namespace +metadata: + name: {{ local_volume_provisioner_namespace }} + labels: + name: {{ local_volume_provisioner_namespace }} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-sa.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-sa.yml.j2 new file mode 100644 index 0000000..c78a16b --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-sa.yml.j2 @@ -0,0 +1,6 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: local-volume-provisioner + namespace: {{ local_volume_provisioner_namespace }} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-sc.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-sc.yml.j2 new file mode 100644 index 0000000..81e0260 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/local_volume_provisioner/templates/local-volume-provisioner-sc.yml.j2 @@ -0,0 +1,12 @@ +{% for class_name, class_config in local_volume_provisioner_storage_classes.items() %} +--- +apiVersion: storage.k8s.io/v1 +kind: StorageClass +metadata: + name: {{ class_name }} +provisioner: kubernetes.io/no-provisioner +volumeBindingMode: WaitForFirstConsumer +{% if class_config.reclaim_policy is defined %} +reclaimPolicy: {{ class_config.reclaim_policy }} +{% endif %} +{% endfor %} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/meta/main.yml b/kubespray/roles/kubernetes-apps/external_provisioner/meta/main.yml new file mode 100644 index 0000000..13bc8b6 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/meta/main.yml @@ -0,0 +1,30 @@ +--- +dependencies: + - role: kubernetes-apps/external_provisioner/local_volume_provisioner + when: + - local_volume_provisioner_enabled + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - apps + - local-volume-provisioner + - external-provisioner + + - role: kubernetes-apps/external_provisioner/cephfs_provisioner + when: cephfs_provisioner_enabled + tags: + - apps + - cephfs-provisioner + - external-provisioner + + - role: kubernetes-apps/external_provisioner/rbd_provisioner + when: rbd_provisioner_enabled + tags: + - apps + - rbd-provisioner + - external-provisioner + - role: kubernetes-apps/external_provisioner/local_path_provisioner + when: local_path_provisioner_enabled + tags: + - apps + - local-path-provisioner + - external-provisioner diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/defaults/main.yml b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/defaults/main.yml new file mode 100644 index 0000000..f09e25a --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/defaults/main.yml @@ -0,0 +1,17 @@ +--- +rbd_provisioner_namespace: "rbd-provisioner" +rbd_provisioner_replicas: 2 +rbd_provisioner_monitors: ~ +rbd_provisioner_pool: kube +rbd_provisioner_admin_id: admin +rbd_provisioner_secret_name: ceph-secret-admin +rbd_provisioner_secret: ceph-key-admin +rbd_provisioner_user_id: kube +rbd_provisioner_user_secret_name: ceph-secret-user +rbd_provisioner_user_secret: ceph-key-user +rbd_provisioner_user_secret_namespace: rbd-provisioner +rbd_provisioner_fs_type: ext4 +rbd_provisioner_image_format: "2" +rbd_provisioner_image_features: layering +rbd_provisioner_storage_class: rbd +rbd_provisioner_reclaim_policy: Delete diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/tasks/main.yml b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/tasks/main.yml new file mode 100644 index 0000000..1d08376 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/tasks/main.yml @@ -0,0 +1,80 @@ +--- + +- name: RBD Provisioner | Remove legacy addon dir and manifests + file: + path: "{{ kube_config_dir }}/addons/rbd_provisioner" + state: absent + when: + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - upgrade + +- name: RBD Provisioner | Remove legacy namespace + command: > + {{ kubectl }} delete namespace {{ rbd_provisioner_namespace }} + ignore_errors: true # noqa ignore-errors + when: + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - upgrade + +- name: RBD Provisioner | Remove legacy storageclass + command: > + {{ kubectl }} delete storageclass {{ rbd_provisioner_storage_class }} + ignore_errors: true # noqa ignore-errors + when: + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - upgrade + +- name: RBD Provisioner | Create addon dir + file: + path: "{{ kube_config_dir }}/addons/rbd_provisioner" + state: directory + owner: root + group: root + mode: 0755 + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: RBD Provisioner | Templates list + set_fact: + rbd_provisioner_templates: + - { name: 00-namespace, file: 00-namespace.yml, type: ns } + - { name: secret-rbd-provisioner, file: secret-rbd-provisioner.yml, type: secret } + - { name: sa-rbd-provisioner, file: sa-rbd-provisioner.yml, type: sa } + - { name: clusterrole-rbd-provisioner, file: clusterrole-rbd-provisioner.yml, type: clusterrole } + - { name: clusterrolebinding-rbd-provisioner, file: clusterrolebinding-rbd-provisioner.yml, type: clusterrolebinding } + - { name: role-rbd-provisioner, file: role-rbd-provisioner.yml, type: role } + - { name: rolebinding-rbd-provisioner, file: rolebinding-rbd-provisioner.yml, type: rolebinding } + - { name: deploy-rbd-provisioner, file: deploy-rbd-provisioner.yml, type: deploy } + - { name: sc-rbd-provisioner, file: sc-rbd-provisioner.yml, type: sc } + rbd_provisioner_templates_for_psp: + - { name: psp-rbd-provisioner, file: psp-rbd-provisioner.yml, type: psp } + +- name: RBD Provisioner | Append extra templates to RBD Provisioner Templates list for PodSecurityPolicy + set_fact: + rbd_provisioner_templates: "{{ rbd_provisioner_templates_for_psp + rbd_provisioner_templates }}" + when: + - podsecuritypolicy_enabled + - rbd_provisioner_namespace != "kube-system" + +- name: RBD Provisioner | Create manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/rbd_provisioner/{{ item.file }}" + mode: 0644 + with_items: "{{ rbd_provisioner_templates }}" + register: rbd_provisioner_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: RBD Provisioner | Apply manifests + kube: + name: "{{ item.item.name }}" + namespace: "{{ rbd_provisioner_namespace }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/rbd_provisioner/{{ item.item.file }}" + state: "latest" + with_items: "{{ rbd_provisioner_manifests.results }}" + when: inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/00-namespace.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/00-namespace.yml.j2 new file mode 100644 index 0000000..8bec2b5 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/00-namespace.yml.j2 @@ -0,0 +1,7 @@ +--- +apiVersion: v1 +kind: Namespace +metadata: + name: {{ rbd_provisioner_namespace }} + labels: + name: {{ rbd_provisioner_namespace }} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/clusterrole-rbd-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/clusterrole-rbd-provisioner.yml.j2 new file mode 100644 index 0000000..8fc7e4b --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/clusterrole-rbd-provisioner.yml.j2 @@ -0,0 +1,30 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: rbd-provisioner + namespace: {{ rbd_provisioner_namespace }} +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch", "create", "delete"] + - apiGroups: [""] + resources: ["persistentvolumeclaims"] + verbs: ["get", "list", "watch", "update"] + - apiGroups: ["storage.k8s.io"] + resources: ["storageclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["create", "update", "patch"] + - apiGroups: [""] + resources: ["services"] + resourceNames: ["kube-dns","coredns"] + verbs: ["list", "get"] + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "create", "delete"] + - apiGroups: ["policy"] + resourceNames: ["rbd-provisioner"] + resources: ["podsecuritypolicies"] + verbs: ["use"] diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/clusterrolebinding-rbd-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/clusterrolebinding-rbd-provisioner.yml.j2 new file mode 100644 index 0000000..ae9e6c5 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/clusterrolebinding-rbd-provisioner.yml.j2 @@ -0,0 +1,13 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: rbd-provisioner +subjects: + - kind: ServiceAccount + name: rbd-provisioner + namespace: {{ rbd_provisioner_namespace }} +roleRef: + kind: ClusterRole + name: rbd-provisioner + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/deploy-rbd-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/deploy-rbd-provisioner.yml.j2 new file mode 100644 index 0000000..b8643db --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/deploy-rbd-provisioner.yml.j2 @@ -0,0 +1,40 @@ +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + name: rbd-provisioner + namespace: {{ rbd_provisioner_namespace }} + labels: + app: rbd-provisioner + version: {{ rbd_provisioner_image_tag }} +spec: + replicas: {{ rbd_provisioner_replicas }} + strategy: + type: Recreate + selector: + matchLabels: + app: rbd-provisioner + version: {{ rbd_provisioner_image_tag }} + template: + metadata: + labels: + app: rbd-provisioner + version: {{ rbd_provisioner_image_tag }} + spec: + priorityClassName: {% if rbd_provisioner_namespace == 'kube-system' %}system-cluster-critical{% else %}k8s-cluster-critical{% endif %}{{ '' }} + serviceAccount: rbd-provisioner + containers: + - name: rbd-provisioner + image: {{ rbd_provisioner_image_repo }}:{{ rbd_provisioner_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + env: + - name: PROVISIONER_NAME + value: ceph.com/rbd + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + command: + - "/usr/local/bin/rbd-provisioner" + args: + - "-id=${POD_NAME}" diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/psp-rbd-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/psp-rbd-provisioner.yml.j2 new file mode 100644 index 0000000..c59effd --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/psp-rbd-provisioner.yml.j2 @@ -0,0 +1,44 @@ +--- +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: rbd-provisioner + annotations: + seccomp.security.alpha.kubernetes.io/defaultProfileName: 'runtime/default' + seccomp.security.alpha.kubernetes.io/allowedProfileNames: 'runtime/default' +{% if apparmor_enabled %} + apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + apparmor.security.beta.kubernetes.io/allowedProfileNames: 'runtime/default' +{% endif %} + labels: + addonmanager.kubernetes.io/mode: Reconcile +spec: + privileged: false + allowPrivilegeEscalation: false + requiredDropCapabilities: + - ALL + volumes: + - 'configMap' + - 'emptyDir' + - 'projected' + - 'secret' + - 'downwardAPI' + - 'persistentVolumeClaim' + hostNetwork: false + hostIPC: false + hostPID: false + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + readOnlyRootFilesystem: false diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/role-rbd-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/role-rbd-provisioner.yml.j2 new file mode 100644 index 0000000..d8dbbf9 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/role-rbd-provisioner.yml.j2 @@ -0,0 +1,13 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: rbd-provisioner + namespace: {{ rbd_provisioner_namespace }} +rules: + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get"] + - apiGroups: [""] + resources: ["endpoints"] + verbs: ["get", "list", "watch", "create", "update", "patch"] diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/rolebinding-rbd-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/rolebinding-rbd-provisioner.yml.j2 new file mode 100644 index 0000000..fcae1cc --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/rolebinding-rbd-provisioner.yml.j2 @@ -0,0 +1,14 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: rbd-provisioner + namespace: {{ rbd_provisioner_namespace }} +subjects: + - kind: ServiceAccount + name: rbd-provisioner + namespace: {{ rbd_provisioner_namespace }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: rbd-provisioner diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/sa-rbd-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/sa-rbd-provisioner.yml.j2 new file mode 100644 index 0000000..c4dce64 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/sa-rbd-provisioner.yml.j2 @@ -0,0 +1,6 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: rbd-provisioner + namespace: {{ rbd_provisioner_namespace }} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/sc-rbd-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/sc-rbd-provisioner.yml.j2 new file mode 100644 index 0000000..9fea17a --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/sc-rbd-provisioner.yml.j2 @@ -0,0 +1,19 @@ +--- +apiVersion: storage.k8s.io/v1 +kind: StorageClass +metadata: + name: {{ rbd_provisioner_storage_class }} +provisioner: ceph.com/rbd +reclaimPolicy: {{ rbd_provisioner_reclaim_policy }} +parameters: + monitors: {{ rbd_provisioner_monitors }} + adminId: {{ rbd_provisioner_admin_id }} + adminSecretNamespace: {{ rbd_provisioner_namespace }} + adminSecretName: {{ rbd_provisioner_secret_name }} + pool: {{ rbd_provisioner_pool }} + userId: {{ rbd_provisioner_user_id }} + userSecretNamespace: {{ rbd_provisioner_user_secret_namespace }} + userSecretName: {{ rbd_provisioner_user_secret_name }} + fsType: "{{ rbd_provisioner_fs_type }}" + imageFormat: "{{ rbd_provisioner_image_format }}" + imageFeatures: {{ rbd_provisioner_image_features }} diff --git a/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/secret-rbd-provisioner.yml.j2 b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/secret-rbd-provisioner.yml.j2 new file mode 100644 index 0000000..a3b66d6 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/external_provisioner/rbd_provisioner/templates/secret-rbd-provisioner.yml.j2 @@ -0,0 +1,18 @@ +--- +kind: Secret +apiVersion: v1 +metadata: + name: {{ rbd_provisioner_secret_name }} + namespace: {{ rbd_provisioner_namespace }} +type: Opaque +data: + secret: {{ rbd_provisioner_secret | b64encode }} +--- +kind: Secret +apiVersion: v1 +metadata: + name: {{ rbd_provisioner_user_secret_name }} + namespace: {{ rbd_provisioner_user_secret_namespace }} +type: Opaque +data: + key: {{ rbd_provisioner_user_secret | b64encode }} diff --git a/kubespray/roles/kubernetes-apps/helm/.gitkeep b/kubespray/roles/kubernetes-apps/helm/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/kubespray/roles/kubernetes-apps/helm/defaults/main.yml b/kubespray/roles/kubernetes-apps/helm/defaults/main.yml new file mode 100644 index 0000000..4dc1cca --- /dev/null +++ b/kubespray/roles/kubernetes-apps/helm/defaults/main.yml @@ -0,0 +1,2 @@ +--- +helm_enabled: false diff --git a/kubespray/roles/kubernetes-apps/helm/tasks/main.yml b/kubespray/roles/kubernetes-apps/helm/tasks/main.yml new file mode 100644 index 0000000..eae0e21 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/helm/tasks/main.yml @@ -0,0 +1,49 @@ +--- +- name: Helm | Gather os specific variables + include_vars: "{{ item }}" + with_first_found: + - files: + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_release }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_major_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}.yml" + - "{{ ansible_os_family | lower }}.yml" + - defaults.yml + paths: + - ../vars + skip: true + +- name: Helm | Install PyYaml + package: + name: "{{ pyyaml_package }}" + state: present + when: pyyaml_package is defined + +- name: Helm | Install PyYaml [flatcar] + include_tasks: pyyaml-flatcar.yml + when: ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + +- name: Helm | Download helm + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.helm) }}" + +- name: Helm | Copy helm binary from download dir + copy: + src: "{{ local_release_dir }}/helm-{{ helm_version }}/linux-{{ image_arch }}/helm" + dest: "{{ bin_dir }}/helm" + mode: 0755 + remote_src: true + +- name: Helm | Get helm completion + command: "{{ bin_dir }}/helm completion bash" + changed_when: False + register: helm_completion + check_mode: False + +- name: Helm | Install helm completion + copy: + dest: /etc/bash_completion.d/helm.sh + content: "{{ helm_completion.stdout }}" + mode: 0755 + become: True diff --git a/kubespray/roles/kubernetes-apps/helm/tasks/pyyaml-flatcar.yml b/kubespray/roles/kubernetes-apps/helm/tasks/pyyaml-flatcar.yml new file mode 100644 index 0000000..ea0d63a --- /dev/null +++ b/kubespray/roles/kubernetes-apps/helm/tasks/pyyaml-flatcar.yml @@ -0,0 +1,22 @@ +--- +- name: Get installed pip version + command: "{{ ansible_python_interpreter if ansible_python_interpreter is defined else 'python' }} -m pip --version" + register: pip_version_output + ignore_errors: yes + changed_when: false + +- name: Get installed PyYAML version + command: "{{ ansible_python_interpreter if ansible_python_interpreter is defined else 'python' }} -m pip show PyYAML" + register: pyyaml_version_output + ignore_errors: yes + changed_when: false + +- name: Install pip + command: "{{ ansible_python_interpreter if ansible_python_interpreter is defined else 'python' }} -m ensurepip --upgrade" + when: (pyyaml_version_output is failed) and (pip_version_output is failed) + +- name: Install PyYAML + ansible.builtin.pip: + name: + - PyYAML + when: (pyyaml_version_output is failed) diff --git a/kubespray/roles/kubernetes-apps/helm/vars/amazon.yml b/kubespray/roles/kubernetes-apps/helm/vars/amazon.yml new file mode 100644 index 0000000..190d262 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/helm/vars/amazon.yml @@ -0,0 +1,2 @@ +--- +pyyaml_package: PyYAML diff --git a/kubespray/roles/kubernetes-apps/helm/vars/centos-7.yml b/kubespray/roles/kubernetes-apps/helm/vars/centos-7.yml new file mode 100644 index 0000000..190d262 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/helm/vars/centos-7.yml @@ -0,0 +1,2 @@ +--- +pyyaml_package: PyYAML diff --git a/kubespray/roles/kubernetes-apps/helm/vars/centos.yml b/kubespray/roles/kubernetes-apps/helm/vars/centos.yml new file mode 100644 index 0000000..ba3964d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/helm/vars/centos.yml @@ -0,0 +1,2 @@ +--- +pyyaml_package: python3-pyyaml diff --git a/kubespray/roles/kubernetes-apps/helm/vars/debian.yml b/kubespray/roles/kubernetes-apps/helm/vars/debian.yml new file mode 100644 index 0000000..db0add5 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/helm/vars/debian.yml @@ -0,0 +1,2 @@ +--- +pyyaml_package: python3-yaml diff --git a/kubespray/roles/kubernetes-apps/helm/vars/fedora.yml b/kubespray/roles/kubernetes-apps/helm/vars/fedora.yml new file mode 100644 index 0000000..ba3964d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/helm/vars/fedora.yml @@ -0,0 +1,2 @@ +--- +pyyaml_package: python3-pyyaml diff --git a/kubespray/roles/kubernetes-apps/helm/vars/redhat-7.yml b/kubespray/roles/kubernetes-apps/helm/vars/redhat-7.yml new file mode 100644 index 0000000..190d262 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/helm/vars/redhat-7.yml @@ -0,0 +1,2 @@ +--- +pyyaml_package: PyYAML diff --git a/kubespray/roles/kubernetes-apps/helm/vars/redhat.yml b/kubespray/roles/kubernetes-apps/helm/vars/redhat.yml new file mode 100644 index 0000000..ba3964d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/helm/vars/redhat.yml @@ -0,0 +1,2 @@ +--- +pyyaml_package: python3-pyyaml diff --git a/kubespray/roles/kubernetes-apps/helm/vars/suse.yml b/kubespray/roles/kubernetes-apps/helm/vars/suse.yml new file mode 100644 index 0000000..ba3964d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/helm/vars/suse.yml @@ -0,0 +1,2 @@ +--- +pyyaml_package: python3-pyyaml diff --git a/kubespray/roles/kubernetes-apps/helm/vars/ubuntu.yml b/kubespray/roles/kubernetes-apps/helm/vars/ubuntu.yml new file mode 100644 index 0000000..db0add5 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/helm/vars/ubuntu.yml @@ -0,0 +1,2 @@ +--- +pyyaml_package: python3-yaml diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/OWNERS b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/OWNERS new file mode 100644 index 0000000..a80e724 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/OWNERS @@ -0,0 +1,6 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +approvers: + - kubespray-approvers +reviewers: + - kubespray-reviewers diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/defaults/main.yml b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/defaults/main.yml new file mode 100644 index 0000000..4c8d97e --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/defaults/main.yml @@ -0,0 +1,7 @@ +--- +alb_ingress_controller_namespace: kube-system +alb_ingress_aws_region: "us-east-1" + +# Enables logging on all outbound requests sent to the AWS API. +# If logging is desired, set to true. +alb_ingress_aws_debug: "false" diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/tasks/main.yml b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/tasks/main.yml new file mode 100644 index 0000000..8a188a4 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/tasks/main.yml @@ -0,0 +1,36 @@ +--- + +- name: ALB Ingress Controller | Create addon dir + file: + path: "{{ kube_config_dir }}/addons/alb_ingress" + state: directory + owner: root + group: root + mode: 0755 + +- name: ALB Ingress Controller | Create manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/alb_ingress/{{ item.file }}" + mode: 0644 + with_items: + - { name: alb-ingress-clusterrole, file: alb-ingress-clusterrole.yml, type: clusterrole } + - { name: alb-ingress-clusterrolebinding, file: alb-ingress-clusterrolebinding.yml, type: clusterrolebinding } + - { name: alb-ingress-ns, file: alb-ingress-ns.yml, type: ns } + - { name: alb-ingress-sa, file: alb-ingress-sa.yml, type: sa } + - { name: alb-ingress-deploy, file: alb-ingress-deploy.yml, type: deploy } + register: alb_ingress_manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: ALB Ingress Controller | Apply manifests + kube: + name: "{{ item.item.name }}" + namespace: "{{ alb_ingress_controller_namespace }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/alb_ingress/{{ item.item.file }}" + state: "latest" + with_items: "{{ alb_ingress_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-clusterrole.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-clusterrole.yml.j2 new file mode 100644 index 0000000..bc03095 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-clusterrole.yml.j2 @@ -0,0 +1,13 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: alb-ingress + namespace: {{ alb_ingress_controller_namespace }} +rules: + - apiGroups: ["", "extensions"] + resources: ["configmaps", "endpoints", "nodes", "pods", "secrets", "events", "ingresses", "ingresses/status", "services"] + verbs: ["list", "create", "get", "update", "watch", "patch"] + - apiGroups: ["", "extensions"] + resources: ["nodes", "pods", "secrets", "services", "namespaces"] + verbs: ["get", "list", "watch"] diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-clusterrolebinding.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-clusterrolebinding.yml.j2 new file mode 100644 index 0000000..71068f4 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-clusterrolebinding.yml.j2 @@ -0,0 +1,14 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: alb-ingress + namespace: {{ alb_ingress_controller_namespace }} +subjects: + - kind: ServiceAccount + name: alb-ingress + namespace: {{ alb_ingress_controller_namespace }} +roleRef: + kind: ClusterRole + name: alb-ingress + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-deploy.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-deploy.yml.j2 new file mode 100644 index 0000000..a3d2834 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-deploy.yml.j2 @@ -0,0 +1,74 @@ +# Application Load Balancer (ALB) Ingress Controller Deployment Manifest. +# This manifest details sensible defaults for deploying an ALB Ingress Controller. +# GitHub: https://github.com/coreos/alb-ingress-controller +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + name: alb-ingress-controller + labels: + k8s-app: alb-ingress-controller + # Namespace the ALB Ingress Controller should run in. Does not impact which + # namespaces it's able to resolve ingress resource for. For limiting ingress + # namespace scope, see --watch-namespace. + namespace: {{ alb_ingress_controller_namespace }} +spec: + replicas: 1 + selector: + matchLabels: + k8s-app: alb-ingress-controller + strategy: + rollingUpdate: + maxSurge: 1 + maxUnavailable: 1 + type: RollingUpdate + template: + metadata: + creationTimestamp: null + labels: + k8s-app: alb-ingress-controller + spec: + containers: + - args: + # Limit the namespace where this ALB Ingress Controller deployment will + # resolve ingress resources. If left commented, all namespaces are used. + #- --watch-namespace=your-k8s-namespace + + # Setting the ingress-class flag below will ensure that only ingress resources with the + # annotation kubernetes.io/ingress.class: "alb" are respected by the controller. You may + # choose any class you'd like for this controller to respect. + - --ingress-class=alb + # Name of your cluster. Used when naming resources created + # by the ALB Ingress Controller, providing distinction between + # clusters. + - --cluster-name={{ cluster_name }} + + # Enables logging on all outbound requests sent to the AWS API. + # If logging is desired, set to true. + # - ---aws-api-debug +{% if alb_ingress_aws_debug %} + - --aws-api-debug +{% endif %} + # Maximum number of times to retry the aws calls. + # defaults to 10. + # - --aws-max-retries=10 + + # AWS region this ingress controller will operate in. + # If unspecified, it will be discovered from ec2metadata. + # List of regions: http://docs.aws.amazon.com/general/latest/gr/rande.html#vpc_region +{% if alb_ingress_aws_region is defined %} + - --aws-region={{ alb_ingress_aws_region }} +{% endif %} + + image: "{{ alb_ingress_image_repo }}:{{ alb_ingress_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + name: server + resources: {} + terminationMessagePath: /dev/termination-log + dnsPolicy: ClusterFirst + restartPolicy: Always + securityContext: {} + terminationGracePeriodSeconds: 30 +{% if rbac_enabled %} + serviceAccountName: alb-ingress +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-ns.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-ns.yml.j2 new file mode 100644 index 0000000..9f57537 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-ns.yml.j2 @@ -0,0 +1,7 @@ +--- +apiVersion: v1 +kind: Namespace +metadata: + name: {{ alb_ingress_controller_namespace }} + labels: + name: {{ alb_ingress_controller_namespace }} diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-sa.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-sa.yml.j2 new file mode 100644 index 0000000..692e3e3 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/alb_ingress_controller/templates/alb-ingress-sa.yml.j2 @@ -0,0 +1,6 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: alb-ingress + namespace: {{ alb_ingress_controller_namespace }} diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/defaults/main.yml b/kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/defaults/main.yml new file mode 100644 index 0000000..0f58bd5 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/defaults/main.yml @@ -0,0 +1,19 @@ +--- +cert_manager_namespace: "cert-manager" +cert_manager_user: 1001 +cert_manager_tolerations: [] +cert_manager_affinity: {} +cert_manager_nodeselector: {} +cert_manager_dns_policy: "ClusterFirst" +cert_manager_dns_config: {} +cert_manager_controller_extra_args: [] + +## Allow http_proxy, https_proxy and no_proxy environment variables +## Details https://github.com/kubernetes-sigs/kubespray/blob/master/docs/proxy.md +cert_manager_http_proxy: "{{ http_proxy | default('') }}" +cert_manager_https_proxy: "{{ https_proxy | default('') }}" +cert_manager_no_proxy: "{{ no_proxy | default('') }}" + +## Change leader election namespace when deploying on GKE Autopilot that forbid the changes on kube-system namespace. +## See https://github.com/jetstack/cert-manager/issues/3717 +cert_manager_leader_election_namespace: kube-system diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/tasks/main.yml b/kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/tasks/main.yml new file mode 100644 index 0000000..4af64ad --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/tasks/main.yml @@ -0,0 +1,56 @@ +--- + +- name: Cert Manager | Remove legacy addon dir and manifests + file: + path: "{{ kube_config_dir }}/addons/cert_manager" + state: absent + when: + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - upgrade + +- name: Cert Manager | Remove legacy namespace + command: > + {{ kubectl }} delete namespace {{ cert_manager_namespace }} + ignore_errors: true # noqa ignore-errors + when: + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - upgrade + +- name: Cert Manager | Create addon dir + file: + path: "{{ kube_config_dir }}/addons/cert_manager" + state: directory + owner: root + group: root + mode: 0755 + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Cert Manager | Templates list + set_fact: + cert_manager_templates: + - { name: cert-manager, file: cert-manager.yml, type: all } + - { name: cert-manager.crds, file: cert-manager.crds.yml, type: crd } + +- name: Cert Manager | Create manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/cert_manager/{{ item.file }}" + mode: 0644 + with_items: "{{ cert_manager_templates }}" + register: cert_manager_manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Cert Manager | Apply manifests + kube: + name: "{{ item.item.name }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/cert_manager/{{ item.item.file }}" + state: "latest" + with_items: "{{ cert_manager_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/templates/cert-manager.crds.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/templates/cert-manager.crds.yml.j2 new file mode 100644 index 0000000..f74ad87 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/templates/cert-manager.crds.yml.j2 @@ -0,0 +1,4422 @@ +# Copyright 2022 The cert-manager Authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +# Source: cert-manager/deploy/crds/crd-clusterissuers.yaml +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: clusterissuers.cert-manager.io + labels: + app: 'cert-manager' + app.kubernetes.io/name: 'cert-manager' + app.kubernetes.io/instance: 'cert-manager' + # Generated labels + app.kubernetes.io/version: "{{ cert_manager_version }}" +spec: + group: cert-manager.io + names: + kind: ClusterIssuer + listKind: ClusterIssuerList + plural: clusterissuers + singular: clusterissuer + categories: + - cert-manager + scope: Cluster + versions: + - name: v1 + subresources: + status: {} + additionalPrinterColumns: + - jsonPath: .status.conditions[?(@.type=="Ready")].status + name: Ready + type: string + - jsonPath: .status.conditions[?(@.type=="Ready")].message + name: Status + priority: 1 + type: string + - jsonPath: .metadata.creationTimestamp + description: CreationTimestamp is a timestamp representing the server time when this object was created. It is not guaranteed to be set in happens-before order across separate operations. Clients may not set this value. It is represented in RFC3339 form and is in UTC. + name: Age + type: date + schema: + openAPIV3Schema: + description: A ClusterIssuer represents a certificate issuing authority which can be referenced as part of `issuerRef` fields. It is similar to an Issuer, however it is cluster-scoped and therefore can be referenced by resources that exist in *any* namespace, not just the same namespace as the referent. + type: object + required: + - spec + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation of an object. Servers should convert recognized schemas to the latest internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this object represents. Servers may infer this from the endpoint the client submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: Desired state of the ClusterIssuer resource. + type: object + properties: + acme: + description: ACME configures this issuer to communicate with a RFC8555 (ACME) server to obtain signed x509 certificates. + type: object + required: + - privateKeySecretRef + - server + properties: + caBundle: + description: Base64-encoded bundle of PEM CAs which can be used to validate the certificate chain presented by the ACME server. Mutually exclusive with SkipTLSVerify; prefer using CABundle to prevent various kinds of security vulnerabilities. If CABundle and SkipTLSVerify are unset, the system certificate bundle inside the container is used to validate the TLS connection. + type: string + format: byte + disableAccountKeyGeneration: + description: Enables or disables generating a new ACME account key. If true, the Issuer resource will *not* request a new account but will expect the account key to be supplied via an existing secret. If false, the cert-manager system will generate a new ACME account key for the Issuer. Defaults to false. + type: boolean + email: + description: Email is the email address to be associated with the ACME account. This field is optional, but it is strongly recommended to be set. It will be used to contact you in case of issues with your account or certificates, including expiry notification emails. This field may be updated after the account is initially registered. + type: string + enableDurationFeature: + description: Enables requesting a Not After date on certificates that matches the duration of the certificate. This is not supported by all ACME servers like Let's Encrypt. If set to true when the ACME server does not support it it will create an error on the Order. Defaults to false. + type: boolean + externalAccountBinding: + description: ExternalAccountBinding is a reference to a CA external account of the ACME server. If set, upon registration cert-manager will attempt to associate the given external account credentials with the registered ACME account. + type: object + required: + - keyID + - keySecretRef + properties: + keyAlgorithm: + description: 'Deprecated: keyAlgorithm field exists for historical compatibility reasons and should not be used. The algorithm is now hardcoded to HS256 in golang/x/crypto/acme.' + type: string + enum: + - HS256 + - HS384 + - HS512 + keyID: + description: keyID is the ID of the CA key that the External Account is bound to. + type: string + keySecretRef: + description: keySecretRef is a Secret Key Selector referencing a data item in a Kubernetes Secret which holds the symmetric MAC key of the External Account Binding. The `key` is the index string that is paired with the key data in the Secret and should not be confused with the key data itself, or indeed with the External Account Binding keyID above. The secret key stored in the Secret **must** be un-padded, base64 URL encoded data. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + preferredChain: + description: 'PreferredChain is the chain to use if the ACME server outputs multiple. PreferredChain is no guarantee that this one gets delivered by the ACME endpoint. For example, for Let''s Encrypt''s DST crosssign you would use: "DST Root CA X3" or "ISRG Root X1" for the newer Let''s Encrypt root CA. This value picks the first certificate bundle in the ACME alternative chains that has a certificate with this value as its issuer''s CN' + type: string + maxLength: 64 + privateKeySecretRef: + description: PrivateKey is the name of a Kubernetes Secret resource that will be used to store the automatically generated ACME account private key. Optionally, a `key` may be specified to select a specific entry within the named Secret resource. If `key` is not specified, a default of `tls.key` will be used. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + server: + description: 'Server is the URL used to access the ACME server''s ''directory'' endpoint. For example, for Let''s Encrypt''s staging endpoint, you would use: "https://acme-staging-v02.api.letsencrypt.org/directory". Only ACME v2 endpoints (i.e. RFC 8555) are supported.' + type: string + skipTLSVerify: + description: 'INSECURE: Enables or disables validation of the ACME server TLS certificate. If true, requests to the ACME server will not have the TLS certificate chain validated. Mutually exclusive with CABundle; prefer using CABundle to prevent various kinds of security vulnerabilities. Only enable this option in development environments. If CABundle and SkipTLSVerify are unset, the system certificate bundle inside the container is used to validate the TLS connection. Defaults to false.' + type: boolean + solvers: + description: 'Solvers is a list of challenge solvers that will be used to solve ACME challenges for the matching domains. Solver configurations must be provided in order to obtain certificates from an ACME server. For more information, see: https://cert-manager.io/docs/configuration/acme/' + type: array + items: + description: An ACMEChallengeSolver describes how to solve ACME challenges for the issuer it is part of. A selector may be provided to use different solving strategies for different DNS names. Only one of HTTP01 or DNS01 must be provided. + type: object + properties: + dns01: + description: Configures cert-manager to attempt to complete authorizations by performing the DNS01 challenge flow. + type: object + properties: + acmeDNS: + description: Use the 'ACME DNS' (https://github.com/joohoi/acme-dns) API to manage DNS01 challenge records. + type: object + required: + - accountSecretRef + - host + properties: + accountSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + host: + type: string + akamai: + description: Use the Akamai DNS zone management API to manage DNS01 challenge records. + type: object + required: + - accessTokenSecretRef + - clientSecretSecretRef + - clientTokenSecretRef + - serviceConsumerDomain + properties: + accessTokenSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + clientSecretSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + clientTokenSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + serviceConsumerDomain: + type: string + azureDNS: + description: Use the Microsoft Azure DNS API to manage DNS01 challenge records. + type: object + required: + - resourceGroupName + - subscriptionID + properties: + clientID: + description: if both this and ClientSecret are left unset MSI will be used + type: string + clientSecretSecretRef: + description: if both this and ClientID are left unset MSI will be used + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + environment: + description: name of the Azure environment (default AzurePublicCloud) + type: string + enum: + - AzurePublicCloud + - AzureChinaCloud + - AzureGermanCloud + - AzureUSGovernmentCloud + hostedZoneName: + description: name of the DNS zone that should be used + type: string + managedIdentity: + description: managed identity configuration, can not be used at the same time as clientID, clientSecretSecretRef or tenantID + type: object + properties: + clientID: + description: client ID of the managed identity, can not be used at the same time as resourceID + type: string + resourceID: + description: resource ID of the managed identity, can not be used at the same time as clientID + type: string + resourceGroupName: + description: resource group the DNS zone is located in + type: string + subscriptionID: + description: ID of the Azure subscription + type: string + tenantID: + description: when specifying ClientID and ClientSecret then this field is also needed + type: string + cloudDNS: + description: Use the Google Cloud DNS API to manage DNS01 challenge records. + type: object + required: + - project + properties: + hostedZoneName: + description: HostedZoneName is an optional field that tells cert-manager in which Cloud DNS zone the challenge record has to be created. If left empty cert-manager will automatically choose a zone. + type: string + project: + type: string + serviceAccountSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + cloudflare: + description: Use the Cloudflare API to manage DNS01 challenge records. + type: object + properties: + apiKeySecretRef: + description: 'API key to use to authenticate with Cloudflare. Note: using an API token to authenticate is now the recommended method as it allows greater control of permissions.' + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + apiTokenSecretRef: + description: API token used to authenticate with Cloudflare. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + email: + description: Email of the account, only required when using API key based authentication. + type: string + cnameStrategy: + description: CNAMEStrategy configures how the DNS01 provider should handle CNAME records when found in DNS zones. + type: string + enum: + - None + - Follow + digitalocean: + description: Use the DigitalOcean DNS API to manage DNS01 challenge records. + type: object + required: + - tokenSecretRef + properties: + tokenSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + rfc2136: + description: Use RFC2136 ("Dynamic Updates in the Domain Name System") (https://datatracker.ietf.org/doc/rfc2136/) to manage DNS01 challenge records. + type: object + required: + - nameserver + properties: + nameserver: + description: The IP address or hostname of an authoritative DNS server supporting RFC2136 in the form host:port. If the host is an IPv6 address it must be enclosed in square brackets (e.g [2001:db8::1]) ; port is optional. This field is required. + type: string + tsigAlgorithm: + description: 'The TSIG Algorithm configured in the DNS supporting RFC2136. Used only when ``tsigSecretSecretRef`` and ``tsigKeyName`` are defined. Supported values are (case-insensitive): ``HMACMD5`` (default), ``HMACSHA1``, ``HMACSHA256`` or ``HMACSHA512``.' + type: string + tsigKeyName: + description: The TSIG Key name configured in the DNS. If ``tsigSecretSecretRef`` is defined, this field is required. + type: string + tsigSecretSecretRef: + description: The name of the secret containing the TSIG value. If ``tsigKeyName`` is defined, this field is required. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + route53: + description: Use the AWS Route53 API to manage DNS01 challenge records. + type: object + required: + - region + properties: + accessKeyID: + description: 'The AccessKeyID is used for authentication. Cannot be set when SecretAccessKeyID is set. If neither the Access Key nor Key ID are set, we fall-back to using env vars, shared credentials file or AWS Instance metadata, see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials' + type: string + accessKeyIDSecretRef: + description: 'The SecretAccessKey is used for authentication. If set, pull the AWS access key ID from a key within a Kubernetes Secret. Cannot be set when AccessKeyID is set. If neither the Access Key nor Key ID are set, we fall-back to using env vars, shared credentials file or AWS Instance metadata, see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials' + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + hostedZoneID: + description: If set, the provider will manage only this zone in Route53 and will not do an lookup using the route53:ListHostedZonesByName api call. + type: string + region: + description: Always set the region when using AccessKeyID and SecretAccessKey + type: string + role: + description: Role is a Role ARN which the Route53 provider will assume using either the explicit credentials AccessKeyID/SecretAccessKey or the inferred credentials from environment variables, shared credentials file or AWS Instance metadata + type: string + secretAccessKeySecretRef: + description: 'The SecretAccessKey is used for authentication. If neither the Access Key nor Key ID are set, we fall-back to using env vars, shared credentials file or AWS Instance metadata, see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials' + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + webhook: + description: Configure an external webhook based DNS01 challenge solver to manage DNS01 challenge records. + type: object + required: + - groupName + - solverName + properties: + config: + description: Additional configuration that should be passed to the webhook apiserver when challenges are processed. This can contain arbitrary JSON data. Secret values should not be specified in this stanza. If secret values are needed (e.g. credentials for a DNS service), you should use a SecretKeySelector to reference a Secret resource. For details on the schema of this field, consult the webhook provider implementation's documentation. + x-kubernetes-preserve-unknown-fields: true + groupName: + description: The API group name that should be used when POSTing ChallengePayload resources to the webhook apiserver. This should be the same as the GroupName specified in the webhook provider implementation. + type: string + solverName: + description: The name of the solver to use, as defined in the webhook provider implementation. This will typically be the name of the provider, e.g. 'cloudflare'. + type: string + http01: + description: Configures cert-manager to attempt to complete authorizations by performing the HTTP01 challenge flow. It is not possible to obtain certificates for wildcard domain names (e.g. `*.example.com`) using the HTTP01 challenge mechanism. + type: object + properties: + gatewayHTTPRoute: + description: The Gateway API is a sig-network community API that models service networking in Kubernetes (https://gateway-api.sigs.k8s.io/). The Gateway solver will create HTTPRoutes with the specified labels in the same namespace as the challenge. This solver is experimental, and fields / behaviour may change in the future. + type: object + properties: + labels: + description: Custom labels that will be applied to HTTPRoutes created by cert-manager while solving HTTP-01 challenges. + type: object + additionalProperties: + type: string + parentRefs: + description: 'When solving an HTTP-01 challenge, cert-manager creates an HTTPRoute. cert-manager needs to know which parentRefs should be used when creating the HTTPRoute. Usually, the parentRef references a Gateway. See: https://gateway-api.sigs.k8s.io/api-types/httproute/#attaching-to-gateways' + type: array + items: + description: "ParentReference identifies an API object (usually a Gateway) that can be considered a parent of this resource (usually a route). The only kind of parent resource with \"Core\" support is Gateway. This API may be extended in the future to support additional kinds of parent resources, such as HTTPRoute. \n The API object must be valid in the cluster; the Group and Kind must be registered in the cluster for this reference to be valid." + type: object + required: + - name + properties: + group: + description: "Group is the group of the referent. When unspecified, \"gateway.networking.k8s.io\" is inferred. To set the core API group (such as for a \"Service\" kind referent), Group must be explicitly set to \"\" (empty string). \n Support: Core" + type: string + default: gateway.networking.k8s.io + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + kind: + description: "Kind is kind of the referent. \n Support: Core (Gateway) \n Support: Implementation-specific (Other Resources)" + type: string + default: Gateway + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + name: + description: "Name is the name of the referent. \n Support: Core" + type: string + maxLength: 253 + minLength: 1 + namespace: + description: "Namespace is the namespace of the referent. When unspecified, this refers to the local namespace of the Route. \n Note that there are specific rules for ParentRefs which cross namespace boundaries. Cross-namespace references are only valid if they are explicitly allowed by something in the namespace they are referring to. For example: Gateway has the AllowedRoutes field, and ReferenceGrant provides a generic way to enable any other kind of cross-namespace reference. \n Support: Core" + type: string + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + port: + description: "Port is the network port this Route targets. It can be interpreted differently based on the type of parent resource. \n When the parent resource is a Gateway, this targets all listeners listening on the specified port that also support this kind of Route(and select this Route). It's not recommended to set `Port` unless the networking behaviors specified in a Route must apply to a specific port as opposed to a listener(s) whose port(s) may be changed. When both Port and SectionName are specified, the name and port of the selected listener must match both specified values. \n Implementations MAY choose to support other parent resources. Implementations supporting other types of parent resources MUST clearly document how/if Port is interpreted. \n For the purpose of status, an attachment is considered successful as long as the parent resource accepts it partially. For example, Gateway listeners can restrict which Routes can attach to them by Route kind, namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from the referencing Route, the Route MUST be considered successfully attached. If no Gateway listeners accept attachment from this Route, the Route MUST be considered detached from the Gateway. \n Support: Extended \n " + type: integer + format: int32 + maximum: 65535 + minimum: 1 + sectionName: + description: "SectionName is the name of a section within the target resource. In the following resources, SectionName is interpreted as the following: \n * Gateway: Listener Name. When both Port (experimental) and SectionName are specified, the name and port of the selected listener must match both specified values. \n Implementations MAY choose to support attaching Routes to other resources. If that is the case, they MUST clearly document how SectionName is interpreted. \n When unspecified (empty string), this will reference the entire resource. For the purpose of status, an attachment is considered successful if at least one section in the parent resource accepts it. For example, Gateway listeners can restrict which Routes can attach to them by Route kind, namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from the referencing Route, the Route MUST be considered successfully attached. If no Gateway listeners accept attachment from this Route, the Route MUST be considered detached from the Gateway. \n Support: Core" + type: string + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + serviceType: + description: Optional service type for Kubernetes solver service. Supported values are NodePort or ClusterIP. If unset, defaults to NodePort. + type: string + ingress: + description: The ingress based HTTP01 challenge solver will solve challenges by creating or modifying Ingress resources in order to route requests for '/.well-known/acme-challenge/XYZ' to 'challenge solver' pods that are provisioned by cert-manager for each Challenge to be completed. + type: object + properties: + class: + description: The ingress class to use when creating Ingress resources to solve ACME challenges that use this challenge solver. Only one of 'class' or 'name' may be specified. + type: string + ingressTemplate: + description: Optional ingress template used to configure the ACME challenge solver ingress used for HTTP01 challenges. + type: object + properties: + metadata: + description: ObjectMeta overrides for the ingress used to solve HTTP01 challenges. Only the 'labels' and 'annotations' fields may be set. If labels or annotations overlap with in-built values, the values here will override the in-built values. + type: object + properties: + annotations: + description: Annotations that should be added to the created ACME HTTP01 solver ingress. + type: object + additionalProperties: + type: string + labels: + description: Labels that should be added to the created ACME HTTP01 solver ingress. + type: object + additionalProperties: + type: string + name: + description: The name of the ingress resource that should have ACME challenge solving routes inserted into it in order to solve HTTP01 challenges. This is typically used in conjunction with ingress controllers like ingress-gce, which maintains a 1:1 mapping between external IPs and ingress resources. + type: string + podTemplate: + description: Optional pod template used to configure the ACME challenge solver pods used for HTTP01 challenges. + type: object + properties: + metadata: + description: ObjectMeta overrides for the pod used to solve HTTP01 challenges. Only the 'labels' and 'annotations' fields may be set. If labels or annotations overlap with in-built values, the values here will override the in-built values. + type: object + properties: + annotations: + description: Annotations that should be added to the create ACME HTTP01 solver pods. + type: object + additionalProperties: + type: string + labels: + description: Labels that should be added to the created ACME HTTP01 solver pods. + type: object + additionalProperties: + type: string + spec: + description: PodSpec defines overrides for the HTTP01 challenge solver pod. Only the 'priorityClassName', 'nodeSelector', 'affinity', 'serviceAccountName' and 'tolerations' fields are supported currently. All other fields will be ignored. + type: object + properties: + affinity: + description: If specified, the pod's scheduling constraints + type: object + properties: + nodeAffinity: + description: Describes node affinity scheduling rules for the pod. + type: object + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: The scheduler will prefer to schedule pods to nodes that satisfy the affinity expressions specified by this field, but it may choose a node that violates one or more of the expressions. The node that is most preferred is the one with the greatest sum of weights, i.e. for each node that meets all of the scheduling requirements (resource request, requiredDuringScheduling affinity expressions, etc.), compute a sum by iterating through the elements of this field and adding "weight" to the sum if the node matches the corresponding matchExpressions; the node(s) with the highest sum are the most preferred. + type: array + items: + description: An empty preferred scheduling term matches all objects with implicit weight 0 (i.e. it's a no-op). A null preferred scheduling term matches no objects (i.e. is also a no-op). + type: object + required: + - preference + - weight + properties: + preference: + description: A node selector term, associated with the corresponding weight. + type: object + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: Represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: An array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. If the operator is Gt or Lt, the values array must have a single element, which will be interpreted as an integer. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchFields: + description: A list of node selector requirements by node's fields. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: Represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: An array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. If the operator is Gt or Lt, the values array must have a single element, which will be interpreted as an integer. This array is replaced during a strategic merge patch. + type: array + items: + type: string + x-kubernetes-map-type: atomic + weight: + description: Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100. + type: integer + format: int32 + requiredDuringSchedulingIgnoredDuringExecution: + description: If the affinity requirements specified by this field are not met at scheduling time, the pod will not be scheduled onto the node. If the affinity requirements specified by this field cease to be met at some point during pod execution (e.g. due to an update), the system may or may not try to eventually evict the pod from its node. + type: object + required: + - nodeSelectorTerms + properties: + nodeSelectorTerms: + description: Required. A list of node selector terms. The terms are ORed. + type: array + items: + description: A null or empty node selector term matches no objects. The requirements of them are ANDed. The TopologySelectorTerm type implements a subset of the NodeSelectorTerm. + type: object + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: Represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: An array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. If the operator is Gt or Lt, the values array must have a single element, which will be interpreted as an integer. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchFields: + description: A list of node selector requirements by node's fields. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: Represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: An array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. If the operator is Gt or Lt, the values array must have a single element, which will be interpreted as an integer. This array is replaced during a strategic merge patch. + type: array + items: + type: string + x-kubernetes-map-type: atomic + x-kubernetes-map-type: atomic + podAffinity: + description: Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)). + type: object + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: The scheduler will prefer to schedule pods to nodes that satisfy the affinity expressions specified by this field, but it may choose a node that violates one or more of the expressions. The node that is most preferred is the one with the greatest sum of weights, i.e. for each node that meets all of the scheduling requirements (resource request, requiredDuringScheduling affinity expressions, etc.), compute a sum by iterating through the elements of this field and adding "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the node(s) with the highest sum are the most preferred. + type: array + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + type: object + required: + - podAffinityTerm + - weight + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaceSelector: + description: A label query over the set of namespaces that the term applies to. The term is applied to the union of the namespaces selected by this field and the ones listed in the namespaces field. null selector and null or empty namespaces list means "this pod's namespace". An empty selector ({}) matches all namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaces: + description: namespaces specifies a static list of namespace names that the term applies to. The term is applied to the union of the namespaces listed in this field and the ones selected by namespaceSelector. null or empty namespaces list and null namespaceSelector means "this pod's namespace". + type: array + items: + type: string + topologyKey: + description: This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching the labelSelector in the specified namespaces, where co-located is defined as running on a node whose value of the label with key topologyKey matches that of any node on which any of the selected pods is running. Empty topologyKey is not allowed. + type: string + weight: + description: weight associated with matching the corresponding podAffinityTerm, in the range 1-100. + type: integer + format: int32 + requiredDuringSchedulingIgnoredDuringExecution: + description: If the affinity requirements specified by this field are not met at scheduling time, the pod will not be scheduled onto the node. If the affinity requirements specified by this field cease to be met at some point during pod execution (e.g. due to a pod label update), the system may or may not try to eventually evict the pod from its node. When there are multiple elements, the lists of nodes corresponding to each podAffinityTerm are intersected, i.e. all terms must be satisfied. + type: array + items: + description: Defines a set of pods (namely those matching the labelSelector relative to the given namespace(s)) that this pod should be co-located (affinity) or not co-located (anti-affinity) with, where co-located is defined as running on a node whose value of the label with key matches that of any node on which a pod of the set of pods is running + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaceSelector: + description: A label query over the set of namespaces that the term applies to. The term is applied to the union of the namespaces selected by this field and the ones listed in the namespaces field. null selector and null or empty namespaces list means "this pod's namespace". An empty selector ({}) matches all namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaces: + description: namespaces specifies a static list of namespace names that the term applies to. The term is applied to the union of the namespaces listed in this field and the ones selected by namespaceSelector. null or empty namespaces list and null namespaceSelector means "this pod's namespace". + type: array + items: + type: string + topologyKey: + description: This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching the labelSelector in the specified namespaces, where co-located is defined as running on a node whose value of the label with key topologyKey matches that of any node on which any of the selected pods is running. Empty topologyKey is not allowed. + type: string + podAntiAffinity: + description: Describes pod anti-affinity scheduling rules (e.g. avoid putting this pod in the same node, zone, etc. as some other pod(s)). + type: object + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: The scheduler will prefer to schedule pods to nodes that satisfy the anti-affinity expressions specified by this field, but it may choose a node that violates one or more of the expressions. The node that is most preferred is the one with the greatest sum of weights, i.e. for each node that meets all of the scheduling requirements (resource request, requiredDuringScheduling anti-affinity expressions, etc.), compute a sum by iterating through the elements of this field and adding "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the node(s) with the highest sum are the most preferred. + type: array + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + type: object + required: + - podAffinityTerm + - weight + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaceSelector: + description: A label query over the set of namespaces that the term applies to. The term is applied to the union of the namespaces selected by this field and the ones listed in the namespaces field. null selector and null or empty namespaces list means "this pod's namespace". An empty selector ({}) matches all namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaces: + description: namespaces specifies a static list of namespace names that the term applies to. The term is applied to the union of the namespaces listed in this field and the ones selected by namespaceSelector. null or empty namespaces list and null namespaceSelector means "this pod's namespace". + type: array + items: + type: string + topologyKey: + description: This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching the labelSelector in the specified namespaces, where co-located is defined as running on a node whose value of the label with key topologyKey matches that of any node on which any of the selected pods is running. Empty topologyKey is not allowed. + type: string + weight: + description: weight associated with matching the corresponding podAffinityTerm, in the range 1-100. + type: integer + format: int32 + requiredDuringSchedulingIgnoredDuringExecution: + description: If the anti-affinity requirements specified by this field are not met at scheduling time, the pod will not be scheduled onto the node. If the anti-affinity requirements specified by this field cease to be met at some point during pod execution (e.g. due to a pod label update), the system may or may not try to eventually evict the pod from its node. When there are multiple elements, the lists of nodes corresponding to each podAffinityTerm are intersected, i.e. all terms must be satisfied. + type: array + items: + description: Defines a set of pods (namely those matching the labelSelector relative to the given namespace(s)) that this pod should be co-located (affinity) or not co-located (anti-affinity) with, where co-located is defined as running on a node whose value of the label with key matches that of any node on which a pod of the set of pods is running + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaceSelector: + description: A label query over the set of namespaces that the term applies to. The term is applied to the union of the namespaces selected by this field and the ones listed in the namespaces field. null selector and null or empty namespaces list means "this pod's namespace". An empty selector ({}) matches all namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaces: + description: namespaces specifies a static list of namespace names that the term applies to. The term is applied to the union of the namespaces listed in this field and the ones selected by namespaceSelector. null or empty namespaces list and null namespaceSelector means "this pod's namespace". + type: array + items: + type: string + topologyKey: + description: This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching the labelSelector in the specified namespaces, where co-located is defined as running on a node whose value of the label with key topologyKey matches that of any node on which any of the selected pods is running. Empty topologyKey is not allowed. + type: string + nodeSelector: + description: 'NodeSelector is a selector which must be true for the pod to fit on a node. Selector which must match a node''s labels for the pod to be scheduled on that node. More info: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/' + type: object + additionalProperties: + type: string + priorityClassName: + description: If specified, the pod's priorityClassName. + type: string + serviceAccountName: + description: If specified, the pod's service account + type: string + tolerations: + description: If specified, the pod's tolerations. + type: array + items: + description: The pod this Toleration is attached to tolerates any taint that matches the triple using the matching operator . + type: object + properties: + effect: + description: Effect indicates the taint effect to match. Empty means match all taint effects. When specified, allowed values are NoSchedule, PreferNoSchedule and NoExecute. + type: string + key: + description: Key is the taint key that the toleration applies to. Empty means match all taint keys. If the key is empty, operator must be Exists; this combination means to match all values and all keys. + type: string + operator: + description: Operator represents a key's relationship to the value. Valid operators are Exists and Equal. Defaults to Equal. Exists is equivalent to wildcard for value, so that a pod can tolerate all taints of a particular category. + type: string + tolerationSeconds: + description: TolerationSeconds represents the period of time the toleration (which must be of effect NoExecute, otherwise this field is ignored) tolerates the taint. By default, it is not set, which means tolerate the taint forever (do not evict). Zero and negative values will be treated as 0 (evict immediately) by the system. + type: integer + format: int64 + value: + description: Value is the taint value the toleration matches to. If the operator is Exists, the value should be empty, otherwise just a regular string. + type: string + serviceType: + description: Optional service type for Kubernetes solver service. Supported values are NodePort or ClusterIP. If unset, defaults to NodePort. + type: string + selector: + description: Selector selects a set of DNSNames on the Certificate resource that should be solved using this challenge solver. If not specified, the solver will be treated as the 'default' solver with the lowest priority, i.e. if any other solver has a more specific match, it will be used instead. + type: object + properties: + dnsNames: + description: List of DNSNames that this solver will be used to solve. If specified and a match is found, a dnsNames selector will take precedence over a dnsZones selector. If multiple solvers match with the same dnsNames value, the solver with the most matching labels in matchLabels will be selected. If neither has more matches, the solver defined earlier in the list will be selected. + type: array + items: + type: string + dnsZones: + description: List of DNSZones that this solver will be used to solve. The most specific DNS zone match specified here will take precedence over other DNS zone matches, so a solver specifying sys.example.com will be selected over one specifying example.com for the domain www.sys.example.com. If multiple solvers match with the same dnsZones value, the solver with the most matching labels in matchLabels will be selected. If neither has more matches, the solver defined earlier in the list will be selected. + type: array + items: + type: string + matchLabels: + description: A label selector that is used to refine the set of certificate's that this challenge solver will apply to. + type: object + additionalProperties: + type: string + ca: + description: CA configures this issuer to sign certificates using a signing CA keypair stored in a Secret resource. This is used to build internal PKIs that are managed by cert-manager. + type: object + required: + - secretName + properties: + crlDistributionPoints: + description: The CRL distribution points is an X.509 v3 certificate extension which identifies the location of the CRL from which the revocation of this certificate can be checked. If not set, certificates will be issued without distribution points set. + type: array + items: + type: string + ocspServers: + description: The OCSP server list is an X.509 v3 extension that defines a list of URLs of OCSP responders. The OCSP responders can be queried for the revocation status of an issued certificate. If not set, the certificate will be issued with no OCSP servers set. For example, an OCSP server URL could be "http://ocsp.int-x3.letsencrypt.org". + type: array + items: + type: string + secretName: + description: SecretName is the name of the secret used to sign Certificates issued by this Issuer. + type: string + selfSigned: + description: SelfSigned configures this issuer to 'self sign' certificates using the private key used to create the CertificateRequest object. + type: object + properties: + crlDistributionPoints: + description: The CRL distribution points is an X.509 v3 certificate extension which identifies the location of the CRL from which the revocation of this certificate can be checked. If not set certificate will be issued without CDP. Values are strings. + type: array + items: + type: string + vault: + description: Vault configures this issuer to sign certificates using a HashiCorp Vault PKI backend. + type: object + required: + - auth + - path + - server + properties: + auth: + description: Auth configures how cert-manager authenticates with the Vault server. + type: object + properties: + appRole: + description: AppRole authenticates with Vault using the App Role auth mechanism, with the role and secret stored in a Kubernetes Secret resource. + type: object + required: + - path + - roleId + - secretRef + properties: + path: + description: 'Path where the App Role authentication backend is mounted in Vault, e.g: "approle"' + type: string + roleId: + description: RoleID configured in the App Role authentication backend when setting up the authentication backend in Vault. + type: string + secretRef: + description: Reference to a key in a Secret that contains the App Role secret used to authenticate with Vault. The `key` field must be specified and denotes which entry within the Secret resource is used as the app role secret. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + kubernetes: + description: Kubernetes authenticates with Vault by passing the ServiceAccount token stored in the named Secret resource to the Vault server. + type: object + required: + - role + - secretRef + properties: + mountPath: + description: The Vault mountPath here is the mount path to use when authenticating with Vault. For example, setting a value to `/v1/auth/foo`, will use the path `/v1/auth/foo/login` to authenticate with Vault. If unspecified, the default value "/v1/auth/kubernetes" will be used. + type: string + role: + description: A required field containing the Vault Role to assume. A Role binds a Kubernetes ServiceAccount with a set of Vault policies. + type: string + secretRef: + description: The required Secret field containing a Kubernetes ServiceAccount JWT used for authenticating with Vault. Use of 'ambient credentials' is not supported. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + tokenSecretRef: + description: TokenSecretRef authenticates with Vault by presenting a token. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + caBundle: + description: Base64-encoded bundle of PEM CAs which will be used to validate the certificate chain presented by Vault. Only used if using HTTPS to connect to Vault and ignored for HTTP connections. Mutually exclusive with CABundleSecretRef. If neither CABundle nor CABundleSecretRef are defined, the certificate bundle in the cert-manager controller container is used to validate the TLS connection. + type: string + format: byte + caBundleSecretRef: + description: Reference to a Secret containing a bundle of PEM-encoded CAs to use when verifying the certificate chain presented by Vault when using HTTPS. Mutually exclusive with CABundle. If neither CABundle nor CABundleSecretRef are defined, the certificate bundle in the cert-manager controller container is used to validate the TLS connection. If no key for the Secret is specified, cert-manager will default to 'ca.crt'. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + namespace: + description: 'Name of the vault namespace. Namespaces is a set of features within Vault Enterprise that allows Vault environments to support Secure Multi-tenancy. e.g: "ns1" More about namespaces can be found here https://www.vaultproject.io/docs/enterprise/namespaces' + type: string + path: + description: 'Path is the mount path of the Vault PKI backend''s `sign` endpoint, e.g: "my_pki_mount/sign/my-role-name".' + type: string + server: + description: 'Server is the connection address for the Vault server, e.g: "https://vault.example.com:8200".' + type: string + venafi: + description: Venafi configures this issuer to sign certificates using a Venafi TPP or Venafi Cloud policy zone. + type: object + required: + - zone + properties: + cloud: + description: Cloud specifies the Venafi cloud configuration settings. Only one of TPP or Cloud may be specified. + type: object + required: + - apiTokenSecretRef + properties: + apiTokenSecretRef: + description: APITokenSecretRef is a secret key selector for the Venafi Cloud API token. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + url: + description: URL is the base URL for Venafi Cloud. Defaults to "https://api.venafi.cloud/v1". + type: string + tpp: + description: TPP specifies Trust Protection Platform configuration settings. Only one of TPP or Cloud may be specified. + type: object + required: + - credentialsRef + - url + properties: + caBundle: + description: Base64-encoded bundle of PEM CAs which will be used to validate the certificate chain presented by the TPP server. Only used if using HTTPS; ignored for HTTP. If undefined, the certificate bundle in the cert-manager controller container is used to validate the chain. + type: string + format: byte + credentialsRef: + description: CredentialsRef is a reference to a Secret containing the username and password for the TPP server. The secret must contain two keys, 'username' and 'password'. + type: object + required: + - name + properties: + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + url: + description: 'URL is the base URL for the vedsdk endpoint of the Venafi TPP instance, for example: "https://tpp.example.com/vedsdk".' + type: string + zone: + description: Zone is the Venafi Policy Zone to use for this issuer. All requests made to the Venafi platform will be restricted by the named zone policy. This field is required. + type: string + status: + description: Status of the ClusterIssuer. This is set and managed automatically. + type: object + properties: + acme: + description: ACME specific status options. This field should only be set if the Issuer is configured to use an ACME server to issue certificates. + type: object + properties: + lastRegisteredEmail: + description: LastRegisteredEmail is the email associated with the latest registered ACME account, in order to track changes made to registered account associated with the Issuer + type: string + uri: + description: URI is the unique account identifier, which can also be used to retrieve account details from the CA + type: string + conditions: + description: List of status conditions to indicate the status of a CertificateRequest. Known condition types are `Ready`. + type: array + items: + description: IssuerCondition contains condition information for an Issuer. + type: object + required: + - status + - type + properties: + lastTransitionTime: + description: LastTransitionTime is the timestamp corresponding to the last status change of this condition. + type: string + format: date-time + message: + description: Message is a human readable description of the details of the last transition, complementing reason. + type: string + observedGeneration: + description: If set, this represents the .metadata.generation that the condition was set based upon. For instance, if .metadata.generation is currently 12, but the .status.condition[x].observedGeneration is 9, the condition is out of date with respect to the current state of the Issuer. + type: integer + format: int64 + reason: + description: Reason is a brief machine readable explanation for the condition's last transition. + type: string + status: + description: Status of the condition, one of (`True`, `False`, `Unknown`). + type: string + enum: + - "True" + - "False" + - Unknown + type: + description: Type of the condition, known values are (`Ready`). + type: string + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + served: true + storage: true +--- +# Source: cert-manager/deploy/crds/crd-challenges.yaml +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: challenges.acme.cert-manager.io + labels: + app: 'cert-manager' + app.kubernetes.io/name: 'cert-manager' + app.kubernetes.io/instance: 'cert-manager' + # Generated labels + app.kubernetes.io/version: "{{ cert_manager_version }}" +spec: + group: acme.cert-manager.io + names: + kind: Challenge + listKind: ChallengeList + plural: challenges + singular: challenge + categories: + - cert-manager + - cert-manager-acme + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .status.state + name: State + type: string + - jsonPath: .spec.dnsName + name: Domain + type: string + - jsonPath: .status.reason + name: Reason + priority: 1 + type: string + - description: CreationTimestamp is a timestamp representing the server time when this object was created. It is not guaranteed to be set in happens-before order across separate operations. Clients may not set this value. It is represented in RFC3339 form and is in UTC. + jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1 + schema: + openAPIV3Schema: + description: Challenge is a type to represent a Challenge request with an ACME server + type: object + required: + - metadata + - spec + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation of an object. Servers should convert recognized schemas to the latest internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this object represents. Servers may infer this from the endpoint the client submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + type: object + required: + - authorizationURL + - dnsName + - issuerRef + - key + - solver + - token + - type + - url + properties: + authorizationURL: + description: The URL to the ACME Authorization resource that this challenge is a part of. + type: string + dnsName: + description: dnsName is the identifier that this challenge is for, e.g. example.com. If the requested DNSName is a 'wildcard', this field MUST be set to the non-wildcard domain, e.g. for `*.example.com`, it must be `example.com`. + type: string + issuerRef: + description: References a properly configured ACME-type Issuer which should be used to create this Challenge. If the Issuer does not exist, processing will be retried. If the Issuer is not an 'ACME' Issuer, an error will be returned and the Challenge will be marked as failed. + type: object + required: + - name + properties: + group: + description: Group of the resource being referred to. + type: string + kind: + description: Kind of the resource being referred to. + type: string + name: + description: Name of the resource being referred to. + type: string + key: + description: 'The ACME challenge key for this challenge For HTTP01 challenges, this is the value that must be responded with to complete the HTTP01 challenge in the format: `.`. For DNS01 challenges, this is the base64 encoded SHA256 sum of the `.` text that must be set as the TXT record content.' + type: string + solver: + description: Contains the domain solving configuration that should be used to solve this challenge resource. + type: object + properties: + dns01: + description: Configures cert-manager to attempt to complete authorizations by performing the DNS01 challenge flow. + type: object + properties: + acmeDNS: + description: Use the 'ACME DNS' (https://github.com/joohoi/acme-dns) API to manage DNS01 challenge records. + type: object + required: + - accountSecretRef + - host + properties: + accountSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + host: + type: string + akamai: + description: Use the Akamai DNS zone management API to manage DNS01 challenge records. + type: object + required: + - accessTokenSecretRef + - clientSecretSecretRef + - clientTokenSecretRef + - serviceConsumerDomain + properties: + accessTokenSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + clientSecretSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + clientTokenSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + serviceConsumerDomain: + type: string + azureDNS: + description: Use the Microsoft Azure DNS API to manage DNS01 challenge records. + type: object + required: + - resourceGroupName + - subscriptionID + properties: + clientID: + description: if both this and ClientSecret are left unset MSI will be used + type: string + clientSecretSecretRef: + description: if both this and ClientID are left unset MSI will be used + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + environment: + description: name of the Azure environment (default AzurePublicCloud) + type: string + enum: + - AzurePublicCloud + - AzureChinaCloud + - AzureGermanCloud + - AzureUSGovernmentCloud + hostedZoneName: + description: name of the DNS zone that should be used + type: string + managedIdentity: + description: managed identity configuration, can not be used at the same time as clientID, clientSecretSecretRef or tenantID + type: object + properties: + clientID: + description: client ID of the managed identity, can not be used at the same time as resourceID + type: string + resourceID: + description: resource ID of the managed identity, can not be used at the same time as clientID + type: string + resourceGroupName: + description: resource group the DNS zone is located in + type: string + subscriptionID: + description: ID of the Azure subscription + type: string + tenantID: + description: when specifying ClientID and ClientSecret then this field is also needed + type: string + cloudDNS: + description: Use the Google Cloud DNS API to manage DNS01 challenge records. + type: object + required: + - project + properties: + hostedZoneName: + description: HostedZoneName is an optional field that tells cert-manager in which Cloud DNS zone the challenge record has to be created. If left empty cert-manager will automatically choose a zone. + type: string + project: + type: string + serviceAccountSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + cloudflare: + description: Use the Cloudflare API to manage DNS01 challenge records. + type: object + properties: + apiKeySecretRef: + description: 'API key to use to authenticate with Cloudflare. Note: using an API token to authenticate is now the recommended method as it allows greater control of permissions.' + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + apiTokenSecretRef: + description: API token used to authenticate with Cloudflare. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + email: + description: Email of the account, only required when using API key based authentication. + type: string + cnameStrategy: + description: CNAMEStrategy configures how the DNS01 provider should handle CNAME records when found in DNS zones. + type: string + enum: + - None + - Follow + digitalocean: + description: Use the DigitalOcean DNS API to manage DNS01 challenge records. + type: object + required: + - tokenSecretRef + properties: + tokenSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + rfc2136: + description: Use RFC2136 ("Dynamic Updates in the Domain Name System") (https://datatracker.ietf.org/doc/rfc2136/) to manage DNS01 challenge records. + type: object + required: + - nameserver + properties: + nameserver: + description: The IP address or hostname of an authoritative DNS server supporting RFC2136 in the form host:port. If the host is an IPv6 address it must be enclosed in square brackets (e.g [2001:db8::1]) ; port is optional. This field is required. + type: string + tsigAlgorithm: + description: 'The TSIG Algorithm configured in the DNS supporting RFC2136. Used only when ``tsigSecretSecretRef`` and ``tsigKeyName`` are defined. Supported values are (case-insensitive): ``HMACMD5`` (default), ``HMACSHA1``, ``HMACSHA256`` or ``HMACSHA512``.' + type: string + tsigKeyName: + description: The TSIG Key name configured in the DNS. If ``tsigSecretSecretRef`` is defined, this field is required. + type: string + tsigSecretSecretRef: + description: The name of the secret containing the TSIG value. If ``tsigKeyName`` is defined, this field is required. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + route53: + description: Use the AWS Route53 API to manage DNS01 challenge records. + type: object + required: + - region + properties: + accessKeyID: + description: 'The AccessKeyID is used for authentication. Cannot be set when SecretAccessKeyID is set. If neither the Access Key nor Key ID are set, we fall-back to using env vars, shared credentials file or AWS Instance metadata, see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials' + type: string + accessKeyIDSecretRef: + description: 'The SecretAccessKey is used for authentication. If set, pull the AWS access key ID from a key within a Kubernetes Secret. Cannot be set when AccessKeyID is set. If neither the Access Key nor Key ID are set, we fall-back to using env vars, shared credentials file or AWS Instance metadata, see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials' + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + hostedZoneID: + description: If set, the provider will manage only this zone in Route53 and will not do an lookup using the route53:ListHostedZonesByName api call. + type: string + region: + description: Always set the region when using AccessKeyID and SecretAccessKey + type: string + role: + description: Role is a Role ARN which the Route53 provider will assume using either the explicit credentials AccessKeyID/SecretAccessKey or the inferred credentials from environment variables, shared credentials file or AWS Instance metadata + type: string + secretAccessKeySecretRef: + description: 'The SecretAccessKey is used for authentication. If neither the Access Key nor Key ID are set, we fall-back to using env vars, shared credentials file or AWS Instance metadata, see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials' + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + webhook: + description: Configure an external webhook based DNS01 challenge solver to manage DNS01 challenge records. + type: object + required: + - groupName + - solverName + properties: + config: + description: Additional configuration that should be passed to the webhook apiserver when challenges are processed. This can contain arbitrary JSON data. Secret values should not be specified in this stanza. If secret values are needed (e.g. credentials for a DNS service), you should use a SecretKeySelector to reference a Secret resource. For details on the schema of this field, consult the webhook provider implementation's documentation. + x-kubernetes-preserve-unknown-fields: true + groupName: + description: The API group name that should be used when POSTing ChallengePayload resources to the webhook apiserver. This should be the same as the GroupName specified in the webhook provider implementation. + type: string + solverName: + description: The name of the solver to use, as defined in the webhook provider implementation. This will typically be the name of the provider, e.g. 'cloudflare'. + type: string + http01: + description: Configures cert-manager to attempt to complete authorizations by performing the HTTP01 challenge flow. It is not possible to obtain certificates for wildcard domain names (e.g. `*.example.com`) using the HTTP01 challenge mechanism. + type: object + properties: + gatewayHTTPRoute: + description: The Gateway API is a sig-network community API that models service networking in Kubernetes (https://gateway-api.sigs.k8s.io/). The Gateway solver will create HTTPRoutes with the specified labels in the same namespace as the challenge. This solver is experimental, and fields / behaviour may change in the future. + type: object + properties: + labels: + description: Custom labels that will be applied to HTTPRoutes created by cert-manager while solving HTTP-01 challenges. + type: object + additionalProperties: + type: string + parentRefs: + description: 'When solving an HTTP-01 challenge, cert-manager creates an HTTPRoute. cert-manager needs to know which parentRefs should be used when creating the HTTPRoute. Usually, the parentRef references a Gateway. See: https://gateway-api.sigs.k8s.io/api-types/httproute/#attaching-to-gateways' + type: array + items: + description: "ParentReference identifies an API object (usually a Gateway) that can be considered a parent of this resource (usually a route). The only kind of parent resource with \"Core\" support is Gateway. This API may be extended in the future to support additional kinds of parent resources, such as HTTPRoute. \n The API object must be valid in the cluster; the Group and Kind must be registered in the cluster for this reference to be valid." + type: object + required: + - name + properties: + group: + description: "Group is the group of the referent. When unspecified, \"gateway.networking.k8s.io\" is inferred. To set the core API group (such as for a \"Service\" kind referent), Group must be explicitly set to \"\" (empty string). \n Support: Core" + type: string + default: gateway.networking.k8s.io + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + kind: + description: "Kind is kind of the referent. \n Support: Core (Gateway) \n Support: Implementation-specific (Other Resources)" + type: string + default: Gateway + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + name: + description: "Name is the name of the referent. \n Support: Core" + type: string + maxLength: 253 + minLength: 1 + namespace: + description: "Namespace is the namespace of the referent. When unspecified, this refers to the local namespace of the Route. \n Note that there are specific rules for ParentRefs which cross namespace boundaries. Cross-namespace references are only valid if they are explicitly allowed by something in the namespace they are referring to. For example: Gateway has the AllowedRoutes field, and ReferenceGrant provides a generic way to enable any other kind of cross-namespace reference. \n Support: Core" + type: string + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + port: + description: "Port is the network port this Route targets. It can be interpreted differently based on the type of parent resource. \n When the parent resource is a Gateway, this targets all listeners listening on the specified port that also support this kind of Route(and select this Route). It's not recommended to set `Port` unless the networking behaviors specified in a Route must apply to a specific port as opposed to a listener(s) whose port(s) may be changed. When both Port and SectionName are specified, the name and port of the selected listener must match both specified values. \n Implementations MAY choose to support other parent resources. Implementations supporting other types of parent resources MUST clearly document how/if Port is interpreted. \n For the purpose of status, an attachment is considered successful as long as the parent resource accepts it partially. For example, Gateway listeners can restrict which Routes can attach to them by Route kind, namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from the referencing Route, the Route MUST be considered successfully attached. If no Gateway listeners accept attachment from this Route, the Route MUST be considered detached from the Gateway. \n Support: Extended \n " + type: integer + format: int32 + maximum: 65535 + minimum: 1 + sectionName: + description: "SectionName is the name of a section within the target resource. In the following resources, SectionName is interpreted as the following: \n * Gateway: Listener Name. When both Port (experimental) and SectionName are specified, the name and port of the selected listener must match both specified values. \n Implementations MAY choose to support attaching Routes to other resources. If that is the case, they MUST clearly document how SectionName is interpreted. \n When unspecified (empty string), this will reference the entire resource. For the purpose of status, an attachment is considered successful if at least one section in the parent resource accepts it. For example, Gateway listeners can restrict which Routes can attach to them by Route kind, namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from the referencing Route, the Route MUST be considered successfully attached. If no Gateway listeners accept attachment from this Route, the Route MUST be considered detached from the Gateway. \n Support: Core" + type: string + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + serviceType: + description: Optional service type for Kubernetes solver service. Supported values are NodePort or ClusterIP. If unset, defaults to NodePort. + type: string + ingress: + description: The ingress based HTTP01 challenge solver will solve challenges by creating or modifying Ingress resources in order to route requests for '/.well-known/acme-challenge/XYZ' to 'challenge solver' pods that are provisioned by cert-manager for each Challenge to be completed. + type: object + properties: + class: + description: The ingress class to use when creating Ingress resources to solve ACME challenges that use this challenge solver. Only one of 'class' or 'name' may be specified. + type: string + ingressTemplate: + description: Optional ingress template used to configure the ACME challenge solver ingress used for HTTP01 challenges. + type: object + properties: + metadata: + description: ObjectMeta overrides for the ingress used to solve HTTP01 challenges. Only the 'labels' and 'annotations' fields may be set. If labels or annotations overlap with in-built values, the values here will override the in-built values. + type: object + properties: + annotations: + description: Annotations that should be added to the created ACME HTTP01 solver ingress. + type: object + additionalProperties: + type: string + labels: + description: Labels that should be added to the created ACME HTTP01 solver ingress. + type: object + additionalProperties: + type: string + name: + description: The name of the ingress resource that should have ACME challenge solving routes inserted into it in order to solve HTTP01 challenges. This is typically used in conjunction with ingress controllers like ingress-gce, which maintains a 1:1 mapping between external IPs and ingress resources. + type: string + podTemplate: + description: Optional pod template used to configure the ACME challenge solver pods used for HTTP01 challenges. + type: object + properties: + metadata: + description: ObjectMeta overrides for the pod used to solve HTTP01 challenges. Only the 'labels' and 'annotations' fields may be set. If labels or annotations overlap with in-built values, the values here will override the in-built values. + type: object + properties: + annotations: + description: Annotations that should be added to the create ACME HTTP01 solver pods. + type: object + additionalProperties: + type: string + labels: + description: Labels that should be added to the created ACME HTTP01 solver pods. + type: object + additionalProperties: + type: string + spec: + description: PodSpec defines overrides for the HTTP01 challenge solver pod. Only the 'priorityClassName', 'nodeSelector', 'affinity', 'serviceAccountName' and 'tolerations' fields are supported currently. All other fields will be ignored. + type: object + properties: + affinity: + description: If specified, the pod's scheduling constraints + type: object + properties: + nodeAffinity: + description: Describes node affinity scheduling rules for the pod. + type: object + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: The scheduler will prefer to schedule pods to nodes that satisfy the affinity expressions specified by this field, but it may choose a node that violates one or more of the expressions. The node that is most preferred is the one with the greatest sum of weights, i.e. for each node that meets all of the scheduling requirements (resource request, requiredDuringScheduling affinity expressions, etc.), compute a sum by iterating through the elements of this field and adding "weight" to the sum if the node matches the corresponding matchExpressions; the node(s) with the highest sum are the most preferred. + type: array + items: + description: An empty preferred scheduling term matches all objects with implicit weight 0 (i.e. it's a no-op). A null preferred scheduling term matches no objects (i.e. is also a no-op). + type: object + required: + - preference + - weight + properties: + preference: + description: A node selector term, associated with the corresponding weight. + type: object + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: Represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: An array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. If the operator is Gt or Lt, the values array must have a single element, which will be interpreted as an integer. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchFields: + description: A list of node selector requirements by node's fields. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: Represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: An array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. If the operator is Gt or Lt, the values array must have a single element, which will be interpreted as an integer. This array is replaced during a strategic merge patch. + type: array + items: + type: string + x-kubernetes-map-type: atomic + weight: + description: Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100. + type: integer + format: int32 + requiredDuringSchedulingIgnoredDuringExecution: + description: If the affinity requirements specified by this field are not met at scheduling time, the pod will not be scheduled onto the node. If the affinity requirements specified by this field cease to be met at some point during pod execution (e.g. due to an update), the system may or may not try to eventually evict the pod from its node. + type: object + required: + - nodeSelectorTerms + properties: + nodeSelectorTerms: + description: Required. A list of node selector terms. The terms are ORed. + type: array + items: + description: A null or empty node selector term matches no objects. The requirements of them are ANDed. The TopologySelectorTerm type implements a subset of the NodeSelectorTerm. + type: object + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: Represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: An array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. If the operator is Gt or Lt, the values array must have a single element, which will be interpreted as an integer. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchFields: + description: A list of node selector requirements by node's fields. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: Represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: An array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. If the operator is Gt or Lt, the values array must have a single element, which will be interpreted as an integer. This array is replaced during a strategic merge patch. + type: array + items: + type: string + x-kubernetes-map-type: atomic + x-kubernetes-map-type: atomic + podAffinity: + description: Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)). + type: object + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: The scheduler will prefer to schedule pods to nodes that satisfy the affinity expressions specified by this field, but it may choose a node that violates one or more of the expressions. The node that is most preferred is the one with the greatest sum of weights, i.e. for each node that meets all of the scheduling requirements (resource request, requiredDuringScheduling affinity expressions, etc.), compute a sum by iterating through the elements of this field and adding "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the node(s) with the highest sum are the most preferred. + type: array + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + type: object + required: + - podAffinityTerm + - weight + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaceSelector: + description: A label query over the set of namespaces that the term applies to. The term is applied to the union of the namespaces selected by this field and the ones listed in the namespaces field. null selector and null or empty namespaces list means "this pod's namespace". An empty selector ({}) matches all namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaces: + description: namespaces specifies a static list of namespace names that the term applies to. The term is applied to the union of the namespaces listed in this field and the ones selected by namespaceSelector. null or empty namespaces list and null namespaceSelector means "this pod's namespace". + type: array + items: + type: string + topologyKey: + description: This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching the labelSelector in the specified namespaces, where co-located is defined as running on a node whose value of the label with key topologyKey matches that of any node on which any of the selected pods is running. Empty topologyKey is not allowed. + type: string + weight: + description: weight associated with matching the corresponding podAffinityTerm, in the range 1-100. + type: integer + format: int32 + requiredDuringSchedulingIgnoredDuringExecution: + description: If the affinity requirements specified by this field are not met at scheduling time, the pod will not be scheduled onto the node. If the affinity requirements specified by this field cease to be met at some point during pod execution (e.g. due to a pod label update), the system may or may not try to eventually evict the pod from its node. When there are multiple elements, the lists of nodes corresponding to each podAffinityTerm are intersected, i.e. all terms must be satisfied. + type: array + items: + description: Defines a set of pods (namely those matching the labelSelector relative to the given namespace(s)) that this pod should be co-located (affinity) or not co-located (anti-affinity) with, where co-located is defined as running on a node whose value of the label with key matches that of any node on which a pod of the set of pods is running + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaceSelector: + description: A label query over the set of namespaces that the term applies to. The term is applied to the union of the namespaces selected by this field and the ones listed in the namespaces field. null selector and null or empty namespaces list means "this pod's namespace". An empty selector ({}) matches all namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaces: + description: namespaces specifies a static list of namespace names that the term applies to. The term is applied to the union of the namespaces listed in this field and the ones selected by namespaceSelector. null or empty namespaces list and null namespaceSelector means "this pod's namespace". + type: array + items: + type: string + topologyKey: + description: This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching the labelSelector in the specified namespaces, where co-located is defined as running on a node whose value of the label with key topologyKey matches that of any node on which any of the selected pods is running. Empty topologyKey is not allowed. + type: string + podAntiAffinity: + description: Describes pod anti-affinity scheduling rules (e.g. avoid putting this pod in the same node, zone, etc. as some other pod(s)). + type: object + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: The scheduler will prefer to schedule pods to nodes that satisfy the anti-affinity expressions specified by this field, but it may choose a node that violates one or more of the expressions. The node that is most preferred is the one with the greatest sum of weights, i.e. for each node that meets all of the scheduling requirements (resource request, requiredDuringScheduling anti-affinity expressions, etc.), compute a sum by iterating through the elements of this field and adding "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the node(s) with the highest sum are the most preferred. + type: array + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + type: object + required: + - podAffinityTerm + - weight + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaceSelector: + description: A label query over the set of namespaces that the term applies to. The term is applied to the union of the namespaces selected by this field and the ones listed in the namespaces field. null selector and null or empty namespaces list means "this pod's namespace". An empty selector ({}) matches all namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaces: + description: namespaces specifies a static list of namespace names that the term applies to. The term is applied to the union of the namespaces listed in this field and the ones selected by namespaceSelector. null or empty namespaces list and null namespaceSelector means "this pod's namespace". + type: array + items: + type: string + topologyKey: + description: This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching the labelSelector in the specified namespaces, where co-located is defined as running on a node whose value of the label with key topologyKey matches that of any node on which any of the selected pods is running. Empty topologyKey is not allowed. + type: string + weight: + description: weight associated with matching the corresponding podAffinityTerm, in the range 1-100. + type: integer + format: int32 + requiredDuringSchedulingIgnoredDuringExecution: + description: If the anti-affinity requirements specified by this field are not met at scheduling time, the pod will not be scheduled onto the node. If the anti-affinity requirements specified by this field cease to be met at some point during pod execution (e.g. due to a pod label update), the system may or may not try to eventually evict the pod from its node. When there are multiple elements, the lists of nodes corresponding to each podAffinityTerm are intersected, i.e. all terms must be satisfied. + type: array + items: + description: Defines a set of pods (namely those matching the labelSelector relative to the given namespace(s)) that this pod should be co-located (affinity) or not co-located (anti-affinity) with, where co-located is defined as running on a node whose value of the label with key matches that of any node on which a pod of the set of pods is running + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaceSelector: + description: A label query over the set of namespaces that the term applies to. The term is applied to the union of the namespaces selected by this field and the ones listed in the namespaces field. null selector and null or empty namespaces list means "this pod's namespace". An empty selector ({}) matches all namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaces: + description: namespaces specifies a static list of namespace names that the term applies to. The term is applied to the union of the namespaces listed in this field and the ones selected by namespaceSelector. null or empty namespaces list and null namespaceSelector means "this pod's namespace". + type: array + items: + type: string + topologyKey: + description: This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching the labelSelector in the specified namespaces, where co-located is defined as running on a node whose value of the label with key topologyKey matches that of any node on which any of the selected pods is running. Empty topologyKey is not allowed. + type: string + nodeSelector: + description: 'NodeSelector is a selector which must be true for the pod to fit on a node. Selector which must match a node''s labels for the pod to be scheduled on that node. More info: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/' + type: object + additionalProperties: + type: string + priorityClassName: + description: If specified, the pod's priorityClassName. + type: string + serviceAccountName: + description: If specified, the pod's service account + type: string + tolerations: + description: If specified, the pod's tolerations. + type: array + items: + description: The pod this Toleration is attached to tolerates any taint that matches the triple using the matching operator . + type: object + properties: + effect: + description: Effect indicates the taint effect to match. Empty means match all taint effects. When specified, allowed values are NoSchedule, PreferNoSchedule and NoExecute. + type: string + key: + description: Key is the taint key that the toleration applies to. Empty means match all taint keys. If the key is empty, operator must be Exists; this combination means to match all values and all keys. + type: string + operator: + description: Operator represents a key's relationship to the value. Valid operators are Exists and Equal. Defaults to Equal. Exists is equivalent to wildcard for value, so that a pod can tolerate all taints of a particular category. + type: string + tolerationSeconds: + description: TolerationSeconds represents the period of time the toleration (which must be of effect NoExecute, otherwise this field is ignored) tolerates the taint. By default, it is not set, which means tolerate the taint forever (do not evict). Zero and negative values will be treated as 0 (evict immediately) by the system. + type: integer + format: int64 + value: + description: Value is the taint value the toleration matches to. If the operator is Exists, the value should be empty, otherwise just a regular string. + type: string + serviceType: + description: Optional service type for Kubernetes solver service. Supported values are NodePort or ClusterIP. If unset, defaults to NodePort. + type: string + selector: + description: Selector selects a set of DNSNames on the Certificate resource that should be solved using this challenge solver. If not specified, the solver will be treated as the 'default' solver with the lowest priority, i.e. if any other solver has a more specific match, it will be used instead. + type: object + properties: + dnsNames: + description: List of DNSNames that this solver will be used to solve. If specified and a match is found, a dnsNames selector will take precedence over a dnsZones selector. If multiple solvers match with the same dnsNames value, the solver with the most matching labels in matchLabels will be selected. If neither has more matches, the solver defined earlier in the list will be selected. + type: array + items: + type: string + dnsZones: + description: List of DNSZones that this solver will be used to solve. The most specific DNS zone match specified here will take precedence over other DNS zone matches, so a solver specifying sys.example.com will be selected over one specifying example.com for the domain www.sys.example.com. If multiple solvers match with the same dnsZones value, the solver with the most matching labels in matchLabels will be selected. If neither has more matches, the solver defined earlier in the list will be selected. + type: array + items: + type: string + matchLabels: + description: A label selector that is used to refine the set of certificate's that this challenge solver will apply to. + type: object + additionalProperties: + type: string + token: + description: The ACME challenge token for this challenge. This is the raw value returned from the ACME server. + type: string + type: + description: The type of ACME challenge this resource represents. One of "HTTP-01" or "DNS-01". + type: string + enum: + - HTTP-01 + - DNS-01 + url: + description: The URL of the ACME Challenge resource for this challenge. This can be used to lookup details about the status of this challenge. + type: string + wildcard: + description: wildcard will be true if this challenge is for a wildcard identifier, for example '*.example.com'. + type: boolean + status: + type: object + properties: + presented: + description: presented will be set to true if the challenge values for this challenge are currently 'presented'. This *does not* imply the self check is passing. Only that the values have been 'submitted' for the appropriate challenge mechanism (i.e. the DNS01 TXT record has been presented, or the HTTP01 configuration has been configured). + type: boolean + processing: + description: Used to denote whether this challenge should be processed or not. This field will only be set to true by the 'scheduling' component. It will only be set to false by the 'challenges' controller, after the challenge has reached a final state or timed out. If this field is set to false, the challenge controller will not take any more action. + type: boolean + reason: + description: Contains human readable information on why the Challenge is in the current state. + type: string + state: + description: Contains the current 'state' of the challenge. If not set, the state of the challenge is unknown. + type: string + enum: + - valid + - ready + - pending + - processing + - invalid + - expired + - errored + served: true + storage: true + subresources: + status: {} +--- +# Source: cert-manager/deploy/crds/crd-certificaterequests.yaml +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: certificaterequests.cert-manager.io + labels: + app: 'cert-manager' + app.kubernetes.io/name: 'cert-manager' + app.kubernetes.io/instance: 'cert-manager' + # Generated labels + app.kubernetes.io/version: "{{ cert_manager_version }}" +spec: + group: cert-manager.io + names: + kind: CertificateRequest + listKind: CertificateRequestList + plural: certificaterequests + shortNames: + - cr + - crs + singular: certificaterequest + categories: + - cert-manager + scope: Namespaced + versions: + - name: v1 + subresources: + status: {} + additionalPrinterColumns: + - jsonPath: .status.conditions[?(@.type=="Approved")].status + name: Approved + type: string + - jsonPath: .status.conditions[?(@.type=="Denied")].status + name: Denied + type: string + - jsonPath: .status.conditions[?(@.type=="Ready")].status + name: Ready + type: string + - jsonPath: .spec.issuerRef.name + name: Issuer + type: string + - jsonPath: .spec.username + name: Requestor + type: string + - jsonPath: .status.conditions[?(@.type=="Ready")].message + name: Status + priority: 1 + type: string + - jsonPath: .metadata.creationTimestamp + description: CreationTimestamp is a timestamp representing the server time when this object was created. It is not guaranteed to be set in happens-before order across separate operations. Clients may not set this value. It is represented in RFC3339 form and is in UTC. + name: Age + type: date + schema: + openAPIV3Schema: + description: "A CertificateRequest is used to request a signed certificate from one of the configured issuers. \n All fields within the CertificateRequest's `spec` are immutable after creation. A CertificateRequest will either succeed or fail, as denoted by its `status.state` field. \n A CertificateRequest is a one-shot resource, meaning it represents a single point in time request for a certificate and cannot be re-used." + type: object + required: + - spec + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation of an object. Servers should convert recognized schemas to the latest internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this object represents. Servers may infer this from the endpoint the client submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: Desired state of the CertificateRequest resource. + type: object + required: + - issuerRef + - request + properties: + duration: + description: The requested 'duration' (i.e. lifetime) of the Certificate. This option may be ignored/overridden by some issuer types. + type: string + extra: + description: Extra contains extra attributes of the user that created the CertificateRequest. Populated by the cert-manager webhook on creation and immutable. + type: object + additionalProperties: + type: array + items: + type: string + groups: + description: Groups contains group membership of the user that created the CertificateRequest. Populated by the cert-manager webhook on creation and immutable. + type: array + items: + type: string + x-kubernetes-list-type: atomic + isCA: + description: IsCA will request to mark the certificate as valid for certificate signing when submitting to the issuer. This will automatically add the `cert sign` usage to the list of `usages`. + type: boolean + issuerRef: + description: IssuerRef is a reference to the issuer for this CertificateRequest. If the `kind` field is not set, or set to `Issuer`, an Issuer resource with the given name in the same namespace as the CertificateRequest will be used. If the `kind` field is set to `ClusterIssuer`, a ClusterIssuer with the provided name will be used. The `name` field in this stanza is required at all times. The group field refers to the API group of the issuer which defaults to `cert-manager.io` if empty. + type: object + required: + - name + properties: + group: + description: Group of the resource being referred to. + type: string + kind: + description: Kind of the resource being referred to. + type: string + name: + description: Name of the resource being referred to. + type: string + request: + description: The PEM-encoded x509 certificate signing request to be submitted to the CA for signing. + type: string + format: byte + uid: + description: UID contains the uid of the user that created the CertificateRequest. Populated by the cert-manager webhook on creation and immutable. + type: string + usages: + description: Usages is the set of x509 usages that are requested for the certificate. If usages are set they SHOULD be encoded inside the CSR spec Defaults to `digital signature` and `key encipherment` if not specified. + type: array + items: + description: "KeyUsage specifies valid usage contexts for keys. See: https://tools.ietf.org/html/rfc5280#section-4.2.1.3 https://tools.ietf.org/html/rfc5280#section-4.2.1.12 \n Valid KeyUsage values are as follows: \"signing\", \"digital signature\", \"content commitment\", \"key encipherment\", \"key agreement\", \"data encipherment\", \"cert sign\", \"crl sign\", \"encipher only\", \"decipher only\", \"any\", \"server auth\", \"client auth\", \"code signing\", \"email protection\", \"s/mime\", \"ipsec end system\", \"ipsec tunnel\", \"ipsec user\", \"timestamping\", \"ocsp signing\", \"microsoft sgc\", \"netscape sgc\"" + type: string + enum: + - signing + - digital signature + - content commitment + - key encipherment + - key agreement + - data encipherment + - cert sign + - crl sign + - encipher only + - decipher only + - any + - server auth + - client auth + - code signing + - email protection + - s/mime + - ipsec end system + - ipsec tunnel + - ipsec user + - timestamping + - ocsp signing + - microsoft sgc + - netscape sgc + username: + description: Username contains the name of the user that created the CertificateRequest. Populated by the cert-manager webhook on creation and immutable. + type: string + status: + description: Status of the CertificateRequest. This is set and managed automatically. + type: object + properties: + ca: + description: The PEM encoded x509 certificate of the signer, also known as the CA (Certificate Authority). This is set on a best-effort basis by different issuers. If not set, the CA is assumed to be unknown/not available. + type: string + format: byte + certificate: + description: The PEM encoded x509 certificate resulting from the certificate signing request. If not set, the CertificateRequest has either not been completed or has failed. More information on failure can be found by checking the `conditions` field. + type: string + format: byte + conditions: + description: List of status conditions to indicate the status of a CertificateRequest. Known condition types are `Ready` and `InvalidRequest`. + type: array + items: + description: CertificateRequestCondition contains condition information for a CertificateRequest. + type: object + required: + - status + - type + properties: + lastTransitionTime: + description: LastTransitionTime is the timestamp corresponding to the last status change of this condition. + type: string + format: date-time + message: + description: Message is a human readable description of the details of the last transition, complementing reason. + type: string + reason: + description: Reason is a brief machine readable explanation for the condition's last transition. + type: string + status: + description: Status of the condition, one of (`True`, `False`, `Unknown`). + type: string + enum: + - "True" + - "False" + - Unknown + type: + description: Type of the condition, known values are (`Ready`, `InvalidRequest`, `Approved`, `Denied`). + type: string + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + failureTime: + description: FailureTime stores the time that this CertificateRequest failed. This is used to influence garbage collection and back-off. + type: string + format: date-time + served: true + storage: true +--- +# Source: cert-manager/deploy/crds/crd-issuers.yaml +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: issuers.cert-manager.io + labels: + app: 'cert-manager' + app.kubernetes.io/name: 'cert-manager' + app.kubernetes.io/instance: 'cert-manager' + # Generated labels + app.kubernetes.io/version: "{{ cert_manager_version }}" +spec: + group: cert-manager.io + names: + kind: Issuer + listKind: IssuerList + plural: issuers + singular: issuer + categories: + - cert-manager + scope: Namespaced + versions: + - name: v1 + subresources: + status: {} + additionalPrinterColumns: + - jsonPath: .status.conditions[?(@.type=="Ready")].status + name: Ready + type: string + - jsonPath: .status.conditions[?(@.type=="Ready")].message + name: Status + priority: 1 + type: string + - jsonPath: .metadata.creationTimestamp + description: CreationTimestamp is a timestamp representing the server time when this object was created. It is not guaranteed to be set in happens-before order across separate operations. Clients may not set this value. It is represented in RFC3339 form and is in UTC. + name: Age + type: date + schema: + openAPIV3Schema: + description: An Issuer represents a certificate issuing authority which can be referenced as part of `issuerRef` fields. It is scoped to a single namespace and can therefore only be referenced by resources within the same namespace. + type: object + required: + - spec + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation of an object. Servers should convert recognized schemas to the latest internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this object represents. Servers may infer this from the endpoint the client submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: Desired state of the Issuer resource. + type: object + properties: + acme: + description: ACME configures this issuer to communicate with a RFC8555 (ACME) server to obtain signed x509 certificates. + type: object + required: + - privateKeySecretRef + - server + properties: + caBundle: + description: Base64-encoded bundle of PEM CAs which can be used to validate the certificate chain presented by the ACME server. Mutually exclusive with SkipTLSVerify; prefer using CABundle to prevent various kinds of security vulnerabilities. If CABundle and SkipTLSVerify are unset, the system certificate bundle inside the container is used to validate the TLS connection. + type: string + format: byte + disableAccountKeyGeneration: + description: Enables or disables generating a new ACME account key. If true, the Issuer resource will *not* request a new account but will expect the account key to be supplied via an existing secret. If false, the cert-manager system will generate a new ACME account key for the Issuer. Defaults to false. + type: boolean + email: + description: Email is the email address to be associated with the ACME account. This field is optional, but it is strongly recommended to be set. It will be used to contact you in case of issues with your account or certificates, including expiry notification emails. This field may be updated after the account is initially registered. + type: string + enableDurationFeature: + description: Enables requesting a Not After date on certificates that matches the duration of the certificate. This is not supported by all ACME servers like Let's Encrypt. If set to true when the ACME server does not support it it will create an error on the Order. Defaults to false. + type: boolean + externalAccountBinding: + description: ExternalAccountBinding is a reference to a CA external account of the ACME server. If set, upon registration cert-manager will attempt to associate the given external account credentials with the registered ACME account. + type: object + required: + - keyID + - keySecretRef + properties: + keyAlgorithm: + description: 'Deprecated: keyAlgorithm field exists for historical compatibility reasons and should not be used. The algorithm is now hardcoded to HS256 in golang/x/crypto/acme.' + type: string + enum: + - HS256 + - HS384 + - HS512 + keyID: + description: keyID is the ID of the CA key that the External Account is bound to. + type: string + keySecretRef: + description: keySecretRef is a Secret Key Selector referencing a data item in a Kubernetes Secret which holds the symmetric MAC key of the External Account Binding. The `key` is the index string that is paired with the key data in the Secret and should not be confused with the key data itself, or indeed with the External Account Binding keyID above. The secret key stored in the Secret **must** be un-padded, base64 URL encoded data. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + preferredChain: + description: 'PreferredChain is the chain to use if the ACME server outputs multiple. PreferredChain is no guarantee that this one gets delivered by the ACME endpoint. For example, for Let''s Encrypt''s DST crosssign you would use: "DST Root CA X3" or "ISRG Root X1" for the newer Let''s Encrypt root CA. This value picks the first certificate bundle in the ACME alternative chains that has a certificate with this value as its issuer''s CN' + type: string + maxLength: 64 + privateKeySecretRef: + description: PrivateKey is the name of a Kubernetes Secret resource that will be used to store the automatically generated ACME account private key. Optionally, a `key` may be specified to select a specific entry within the named Secret resource. If `key` is not specified, a default of `tls.key` will be used. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + server: + description: 'Server is the URL used to access the ACME server''s ''directory'' endpoint. For example, for Let''s Encrypt''s staging endpoint, you would use: "https://acme-staging-v02.api.letsencrypt.org/directory". Only ACME v2 endpoints (i.e. RFC 8555) are supported.' + type: string + skipTLSVerify: + description: 'INSECURE: Enables or disables validation of the ACME server TLS certificate. If true, requests to the ACME server will not have the TLS certificate chain validated. Mutually exclusive with CABundle; prefer using CABundle to prevent various kinds of security vulnerabilities. Only enable this option in development environments. If CABundle and SkipTLSVerify are unset, the system certificate bundle inside the container is used to validate the TLS connection. Defaults to false.' + type: boolean + solvers: + description: 'Solvers is a list of challenge solvers that will be used to solve ACME challenges for the matching domains. Solver configurations must be provided in order to obtain certificates from an ACME server. For more information, see: https://cert-manager.io/docs/configuration/acme/' + type: array + items: + description: An ACMEChallengeSolver describes how to solve ACME challenges for the issuer it is part of. A selector may be provided to use different solving strategies for different DNS names. Only one of HTTP01 or DNS01 must be provided. + type: object + properties: + dns01: + description: Configures cert-manager to attempt to complete authorizations by performing the DNS01 challenge flow. + type: object + properties: + acmeDNS: + description: Use the 'ACME DNS' (https://github.com/joohoi/acme-dns) API to manage DNS01 challenge records. + type: object + required: + - accountSecretRef + - host + properties: + accountSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + host: + type: string + akamai: + description: Use the Akamai DNS zone management API to manage DNS01 challenge records. + type: object + required: + - accessTokenSecretRef + - clientSecretSecretRef + - clientTokenSecretRef + - serviceConsumerDomain + properties: + accessTokenSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + clientSecretSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + clientTokenSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + serviceConsumerDomain: + type: string + azureDNS: + description: Use the Microsoft Azure DNS API to manage DNS01 challenge records. + type: object + required: + - resourceGroupName + - subscriptionID + properties: + clientID: + description: if both this and ClientSecret are left unset MSI will be used + type: string + clientSecretSecretRef: + description: if both this and ClientID are left unset MSI will be used + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + environment: + description: name of the Azure environment (default AzurePublicCloud) + type: string + enum: + - AzurePublicCloud + - AzureChinaCloud + - AzureGermanCloud + - AzureUSGovernmentCloud + hostedZoneName: + description: name of the DNS zone that should be used + type: string + managedIdentity: + description: managed identity configuration, can not be used at the same time as clientID, clientSecretSecretRef or tenantID + type: object + properties: + clientID: + description: client ID of the managed identity, can not be used at the same time as resourceID + type: string + resourceID: + description: resource ID of the managed identity, can not be used at the same time as clientID + type: string + resourceGroupName: + description: resource group the DNS zone is located in + type: string + subscriptionID: + description: ID of the Azure subscription + type: string + tenantID: + description: when specifying ClientID and ClientSecret then this field is also needed + type: string + cloudDNS: + description: Use the Google Cloud DNS API to manage DNS01 challenge records. + type: object + required: + - project + properties: + hostedZoneName: + description: HostedZoneName is an optional field that tells cert-manager in which Cloud DNS zone the challenge record has to be created. If left empty cert-manager will automatically choose a zone. + type: string + project: + type: string + serviceAccountSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + cloudflare: + description: Use the Cloudflare API to manage DNS01 challenge records. + type: object + properties: + apiKeySecretRef: + description: 'API key to use to authenticate with Cloudflare. Note: using an API token to authenticate is now the recommended method as it allows greater control of permissions.' + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + apiTokenSecretRef: + description: API token used to authenticate with Cloudflare. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + email: + description: Email of the account, only required when using API key based authentication. + type: string + cnameStrategy: + description: CNAMEStrategy configures how the DNS01 provider should handle CNAME records when found in DNS zones. + type: string + enum: + - None + - Follow + digitalocean: + description: Use the DigitalOcean DNS API to manage DNS01 challenge records. + type: object + required: + - tokenSecretRef + properties: + tokenSecretRef: + description: A reference to a specific 'key' within a Secret resource. In some instances, `key` is a required field. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + rfc2136: + description: Use RFC2136 ("Dynamic Updates in the Domain Name System") (https://datatracker.ietf.org/doc/rfc2136/) to manage DNS01 challenge records. + type: object + required: + - nameserver + properties: + nameserver: + description: The IP address or hostname of an authoritative DNS server supporting RFC2136 in the form host:port. If the host is an IPv6 address it must be enclosed in square brackets (e.g [2001:db8::1]) ; port is optional. This field is required. + type: string + tsigAlgorithm: + description: 'The TSIG Algorithm configured in the DNS supporting RFC2136. Used only when ``tsigSecretSecretRef`` and ``tsigKeyName`` are defined. Supported values are (case-insensitive): ``HMACMD5`` (default), ``HMACSHA1``, ``HMACSHA256`` or ``HMACSHA512``.' + type: string + tsigKeyName: + description: The TSIG Key name configured in the DNS. If ``tsigSecretSecretRef`` is defined, this field is required. + type: string + tsigSecretSecretRef: + description: The name of the secret containing the TSIG value. If ``tsigKeyName`` is defined, this field is required. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + route53: + description: Use the AWS Route53 API to manage DNS01 challenge records. + type: object + required: + - region + properties: + accessKeyID: + description: 'The AccessKeyID is used for authentication. Cannot be set when SecretAccessKeyID is set. If neither the Access Key nor Key ID are set, we fall-back to using env vars, shared credentials file or AWS Instance metadata, see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials' + type: string + accessKeyIDSecretRef: + description: 'The SecretAccessKey is used for authentication. If set, pull the AWS access key ID from a key within a Kubernetes Secret. Cannot be set when AccessKeyID is set. If neither the Access Key nor Key ID are set, we fall-back to using env vars, shared credentials file or AWS Instance metadata, see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials' + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + hostedZoneID: + description: If set, the provider will manage only this zone in Route53 and will not do an lookup using the route53:ListHostedZonesByName api call. + type: string + region: + description: Always set the region when using AccessKeyID and SecretAccessKey + type: string + role: + description: Role is a Role ARN which the Route53 provider will assume using either the explicit credentials AccessKeyID/SecretAccessKey or the inferred credentials from environment variables, shared credentials file or AWS Instance metadata + type: string + secretAccessKeySecretRef: + description: 'The SecretAccessKey is used for authentication. If neither the Access Key nor Key ID are set, we fall-back to using env vars, shared credentials file or AWS Instance metadata, see: https://docs.aws.amazon.com/sdk-for-go/v1/developer-guide/configuring-sdk.html#specifying-credentials' + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + webhook: + description: Configure an external webhook based DNS01 challenge solver to manage DNS01 challenge records. + type: object + required: + - groupName + - solverName + properties: + config: + description: Additional configuration that should be passed to the webhook apiserver when challenges are processed. This can contain arbitrary JSON data. Secret values should not be specified in this stanza. If secret values are needed (e.g. credentials for a DNS service), you should use a SecretKeySelector to reference a Secret resource. For details on the schema of this field, consult the webhook provider implementation's documentation. + x-kubernetes-preserve-unknown-fields: true + groupName: + description: The API group name that should be used when POSTing ChallengePayload resources to the webhook apiserver. This should be the same as the GroupName specified in the webhook provider implementation. + type: string + solverName: + description: The name of the solver to use, as defined in the webhook provider implementation. This will typically be the name of the provider, e.g. 'cloudflare'. + type: string + http01: + description: Configures cert-manager to attempt to complete authorizations by performing the HTTP01 challenge flow. It is not possible to obtain certificates for wildcard domain names (e.g. `*.example.com`) using the HTTP01 challenge mechanism. + type: object + properties: + gatewayHTTPRoute: + description: The Gateway API is a sig-network community API that models service networking in Kubernetes (https://gateway-api.sigs.k8s.io/). The Gateway solver will create HTTPRoutes with the specified labels in the same namespace as the challenge. This solver is experimental, and fields / behaviour may change in the future. + type: object + properties: + labels: + description: Custom labels that will be applied to HTTPRoutes created by cert-manager while solving HTTP-01 challenges. + type: object + additionalProperties: + type: string + parentRefs: + description: 'When solving an HTTP-01 challenge, cert-manager creates an HTTPRoute. cert-manager needs to know which parentRefs should be used when creating the HTTPRoute. Usually, the parentRef references a Gateway. See: https://gateway-api.sigs.k8s.io/api-types/httproute/#attaching-to-gateways' + type: array + items: + description: "ParentReference identifies an API object (usually a Gateway) that can be considered a parent of this resource (usually a route). The only kind of parent resource with \"Core\" support is Gateway. This API may be extended in the future to support additional kinds of parent resources, such as HTTPRoute. \n The API object must be valid in the cluster; the Group and Kind must be registered in the cluster for this reference to be valid." + type: object + required: + - name + properties: + group: + description: "Group is the group of the referent. When unspecified, \"gateway.networking.k8s.io\" is inferred. To set the core API group (such as for a \"Service\" kind referent), Group must be explicitly set to \"\" (empty string). \n Support: Core" + type: string + default: gateway.networking.k8s.io + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + kind: + description: "Kind is kind of the referent. \n Support: Core (Gateway) \n Support: Implementation-specific (Other Resources)" + type: string + default: Gateway + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + name: + description: "Name is the name of the referent. \n Support: Core" + type: string + maxLength: 253 + minLength: 1 + namespace: + description: "Namespace is the namespace of the referent. When unspecified, this refers to the local namespace of the Route. \n Note that there are specific rules for ParentRefs which cross namespace boundaries. Cross-namespace references are only valid if they are explicitly allowed by something in the namespace they are referring to. For example: Gateway has the AllowedRoutes field, and ReferenceGrant provides a generic way to enable any other kind of cross-namespace reference. \n Support: Core" + type: string + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + port: + description: "Port is the network port this Route targets. It can be interpreted differently based on the type of parent resource. \n When the parent resource is a Gateway, this targets all listeners listening on the specified port that also support this kind of Route(and select this Route). It's not recommended to set `Port` unless the networking behaviors specified in a Route must apply to a specific port as opposed to a listener(s) whose port(s) may be changed. When both Port and SectionName are specified, the name and port of the selected listener must match both specified values. \n Implementations MAY choose to support other parent resources. Implementations supporting other types of parent resources MUST clearly document how/if Port is interpreted. \n For the purpose of status, an attachment is considered successful as long as the parent resource accepts it partially. For example, Gateway listeners can restrict which Routes can attach to them by Route kind, namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from the referencing Route, the Route MUST be considered successfully attached. If no Gateway listeners accept attachment from this Route, the Route MUST be considered detached from the Gateway. \n Support: Extended \n " + type: integer + format: int32 + maximum: 65535 + minimum: 1 + sectionName: + description: "SectionName is the name of a section within the target resource. In the following resources, SectionName is interpreted as the following: \n * Gateway: Listener Name. When both Port (experimental) and SectionName are specified, the name and port of the selected listener must match both specified values. \n Implementations MAY choose to support attaching Routes to other resources. If that is the case, they MUST clearly document how SectionName is interpreted. \n When unspecified (empty string), this will reference the entire resource. For the purpose of status, an attachment is considered successful if at least one section in the parent resource accepts it. For example, Gateway listeners can restrict which Routes can attach to them by Route kind, namespace, or hostname. If 1 of 2 Gateway listeners accept attachment from the referencing Route, the Route MUST be considered successfully attached. If no Gateway listeners accept attachment from this Route, the Route MUST be considered detached from the Gateway. \n Support: Core" + type: string + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + serviceType: + description: Optional service type for Kubernetes solver service. Supported values are NodePort or ClusterIP. If unset, defaults to NodePort. + type: string + ingress: + description: The ingress based HTTP01 challenge solver will solve challenges by creating or modifying Ingress resources in order to route requests for '/.well-known/acme-challenge/XYZ' to 'challenge solver' pods that are provisioned by cert-manager for each Challenge to be completed. + type: object + properties: + class: + description: The ingress class to use when creating Ingress resources to solve ACME challenges that use this challenge solver. Only one of 'class' or 'name' may be specified. + type: string + ingressTemplate: + description: Optional ingress template used to configure the ACME challenge solver ingress used for HTTP01 challenges. + type: object + properties: + metadata: + description: ObjectMeta overrides for the ingress used to solve HTTP01 challenges. Only the 'labels' and 'annotations' fields may be set. If labels or annotations overlap with in-built values, the values here will override the in-built values. + type: object + properties: + annotations: + description: Annotations that should be added to the created ACME HTTP01 solver ingress. + type: object + additionalProperties: + type: string + labels: + description: Labels that should be added to the created ACME HTTP01 solver ingress. + type: object + additionalProperties: + type: string + name: + description: The name of the ingress resource that should have ACME challenge solving routes inserted into it in order to solve HTTP01 challenges. This is typically used in conjunction with ingress controllers like ingress-gce, which maintains a 1:1 mapping between external IPs and ingress resources. + type: string + podTemplate: + description: Optional pod template used to configure the ACME challenge solver pods used for HTTP01 challenges. + type: object + properties: + metadata: + description: ObjectMeta overrides for the pod used to solve HTTP01 challenges. Only the 'labels' and 'annotations' fields may be set. If labels or annotations overlap with in-built values, the values here will override the in-built values. + type: object + properties: + annotations: + description: Annotations that should be added to the create ACME HTTP01 solver pods. + type: object + additionalProperties: + type: string + labels: + description: Labels that should be added to the created ACME HTTP01 solver pods. + type: object + additionalProperties: + type: string + spec: + description: PodSpec defines overrides for the HTTP01 challenge solver pod. Only the 'priorityClassName', 'nodeSelector', 'affinity', 'serviceAccountName' and 'tolerations' fields are supported currently. All other fields will be ignored. + type: object + properties: + affinity: + description: If specified, the pod's scheduling constraints + type: object + properties: + nodeAffinity: + description: Describes node affinity scheduling rules for the pod. + type: object + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: The scheduler will prefer to schedule pods to nodes that satisfy the affinity expressions specified by this field, but it may choose a node that violates one or more of the expressions. The node that is most preferred is the one with the greatest sum of weights, i.e. for each node that meets all of the scheduling requirements (resource request, requiredDuringScheduling affinity expressions, etc.), compute a sum by iterating through the elements of this field and adding "weight" to the sum if the node matches the corresponding matchExpressions; the node(s) with the highest sum are the most preferred. + type: array + items: + description: An empty preferred scheduling term matches all objects with implicit weight 0 (i.e. it's a no-op). A null preferred scheduling term matches no objects (i.e. is also a no-op). + type: object + required: + - preference + - weight + properties: + preference: + description: A node selector term, associated with the corresponding weight. + type: object + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: Represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: An array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. If the operator is Gt or Lt, the values array must have a single element, which will be interpreted as an integer. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchFields: + description: A list of node selector requirements by node's fields. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: Represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: An array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. If the operator is Gt or Lt, the values array must have a single element, which will be interpreted as an integer. This array is replaced during a strategic merge patch. + type: array + items: + type: string + x-kubernetes-map-type: atomic + weight: + description: Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100. + type: integer + format: int32 + requiredDuringSchedulingIgnoredDuringExecution: + description: If the affinity requirements specified by this field are not met at scheduling time, the pod will not be scheduled onto the node. If the affinity requirements specified by this field cease to be met at some point during pod execution (e.g. due to an update), the system may or may not try to eventually evict the pod from its node. + type: object + required: + - nodeSelectorTerms + properties: + nodeSelectorTerms: + description: Required. A list of node selector terms. The terms are ORed. + type: array + items: + description: A null or empty node selector term matches no objects. The requirements of them are ANDed. The TopologySelectorTerm type implements a subset of the NodeSelectorTerm. + type: object + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: Represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: An array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. If the operator is Gt or Lt, the values array must have a single element, which will be interpreted as an integer. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchFields: + description: A list of node selector requirements by node's fields. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: Represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: An array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. If the operator is Gt or Lt, the values array must have a single element, which will be interpreted as an integer. This array is replaced during a strategic merge patch. + type: array + items: + type: string + x-kubernetes-map-type: atomic + x-kubernetes-map-type: atomic + podAffinity: + description: Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)). + type: object + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: The scheduler will prefer to schedule pods to nodes that satisfy the affinity expressions specified by this field, but it may choose a node that violates one or more of the expressions. The node that is most preferred is the one with the greatest sum of weights, i.e. for each node that meets all of the scheduling requirements (resource request, requiredDuringScheduling affinity expressions, etc.), compute a sum by iterating through the elements of this field and adding "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the node(s) with the highest sum are the most preferred. + type: array + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + type: object + required: + - podAffinityTerm + - weight + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaceSelector: + description: A label query over the set of namespaces that the term applies to. The term is applied to the union of the namespaces selected by this field and the ones listed in the namespaces field. null selector and null or empty namespaces list means "this pod's namespace". An empty selector ({}) matches all namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaces: + description: namespaces specifies a static list of namespace names that the term applies to. The term is applied to the union of the namespaces listed in this field and the ones selected by namespaceSelector. null or empty namespaces list and null namespaceSelector means "this pod's namespace". + type: array + items: + type: string + topologyKey: + description: This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching the labelSelector in the specified namespaces, where co-located is defined as running on a node whose value of the label with key topologyKey matches that of any node on which any of the selected pods is running. Empty topologyKey is not allowed. + type: string + weight: + description: weight associated with matching the corresponding podAffinityTerm, in the range 1-100. + type: integer + format: int32 + requiredDuringSchedulingIgnoredDuringExecution: + description: If the affinity requirements specified by this field are not met at scheduling time, the pod will not be scheduled onto the node. If the affinity requirements specified by this field cease to be met at some point during pod execution (e.g. due to a pod label update), the system may or may not try to eventually evict the pod from its node. When there are multiple elements, the lists of nodes corresponding to each podAffinityTerm are intersected, i.e. all terms must be satisfied. + type: array + items: + description: Defines a set of pods (namely those matching the labelSelector relative to the given namespace(s)) that this pod should be co-located (affinity) or not co-located (anti-affinity) with, where co-located is defined as running on a node whose value of the label with key matches that of any node on which a pod of the set of pods is running + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaceSelector: + description: A label query over the set of namespaces that the term applies to. The term is applied to the union of the namespaces selected by this field and the ones listed in the namespaces field. null selector and null or empty namespaces list means "this pod's namespace". An empty selector ({}) matches all namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaces: + description: namespaces specifies a static list of namespace names that the term applies to. The term is applied to the union of the namespaces listed in this field and the ones selected by namespaceSelector. null or empty namespaces list and null namespaceSelector means "this pod's namespace". + type: array + items: + type: string + topologyKey: + description: This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching the labelSelector in the specified namespaces, where co-located is defined as running on a node whose value of the label with key topologyKey matches that of any node on which any of the selected pods is running. Empty topologyKey is not allowed. + type: string + podAntiAffinity: + description: Describes pod anti-affinity scheduling rules (e.g. avoid putting this pod in the same node, zone, etc. as some other pod(s)). + type: object + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: The scheduler will prefer to schedule pods to nodes that satisfy the anti-affinity expressions specified by this field, but it may choose a node that violates one or more of the expressions. The node that is most preferred is the one with the greatest sum of weights, i.e. for each node that meets all of the scheduling requirements (resource request, requiredDuringScheduling anti-affinity expressions, etc.), compute a sum by iterating through the elements of this field and adding "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the node(s) with the highest sum are the most preferred. + type: array + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + type: object + required: + - podAffinityTerm + - weight + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaceSelector: + description: A label query over the set of namespaces that the term applies to. The term is applied to the union of the namespaces selected by this field and the ones listed in the namespaces field. null selector and null or empty namespaces list means "this pod's namespace". An empty selector ({}) matches all namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaces: + description: namespaces specifies a static list of namespace names that the term applies to. The term is applied to the union of the namespaces listed in this field and the ones selected by namespaceSelector. null or empty namespaces list and null namespaceSelector means "this pod's namespace". + type: array + items: + type: string + topologyKey: + description: This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching the labelSelector in the specified namespaces, where co-located is defined as running on a node whose value of the label with key topologyKey matches that of any node on which any of the selected pods is running. Empty topologyKey is not allowed. + type: string + weight: + description: weight associated with matching the corresponding podAffinityTerm, in the range 1-100. + type: integer + format: int32 + requiredDuringSchedulingIgnoredDuringExecution: + description: If the anti-affinity requirements specified by this field are not met at scheduling time, the pod will not be scheduled onto the node. If the anti-affinity requirements specified by this field cease to be met at some point during pod execution (e.g. due to a pod label update), the system may or may not try to eventually evict the pod from its node. When there are multiple elements, the lists of nodes corresponding to each podAffinityTerm are intersected, i.e. all terms must be satisfied. + type: array + items: + description: Defines a set of pods (namely those matching the labelSelector relative to the given namespace(s)) that this pod should be co-located (affinity) or not co-located (anti-affinity) with, where co-located is defined as running on a node whose value of the label with key matches that of any node on which a pod of the set of pods is running + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaceSelector: + description: A label query over the set of namespaces that the term applies to. The term is applied to the union of the namespaces selected by this field and the ones listed in the namespaces field. null selector and null or empty namespaces list means "this pod's namespace". An empty selector ({}) matches all namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: operator represents a key's relationship to a set of values. Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. If the operator is In or NotIn, the values array must be non-empty. If the operator is Exists or DoesNotExist, the values array must be empty. This array is replaced during a strategic merge patch. + type: array + items: + type: string + matchLabels: + description: matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels map is equivalent to an element of matchExpressions, whose key field is "key", the operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + namespaces: + description: namespaces specifies a static list of namespace names that the term applies to. The term is applied to the union of the namespaces listed in this field and the ones selected by namespaceSelector. null or empty namespaces list and null namespaceSelector means "this pod's namespace". + type: array + items: + type: string + topologyKey: + description: This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching the labelSelector in the specified namespaces, where co-located is defined as running on a node whose value of the label with key topologyKey matches that of any node on which any of the selected pods is running. Empty topologyKey is not allowed. + type: string + nodeSelector: + description: 'NodeSelector is a selector which must be true for the pod to fit on a node. Selector which must match a node''s labels for the pod to be scheduled on that node. More info: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/' + type: object + additionalProperties: + type: string + priorityClassName: + description: If specified, the pod's priorityClassName. + type: string + serviceAccountName: + description: If specified, the pod's service account + type: string + tolerations: + description: If specified, the pod's tolerations. + type: array + items: + description: The pod this Toleration is attached to tolerates any taint that matches the triple using the matching operator . + type: object + properties: + effect: + description: Effect indicates the taint effect to match. Empty means match all taint effects. When specified, allowed values are NoSchedule, PreferNoSchedule and NoExecute. + type: string + key: + description: Key is the taint key that the toleration applies to. Empty means match all taint keys. If the key is empty, operator must be Exists; this combination means to match all values and all keys. + type: string + operator: + description: Operator represents a key's relationship to the value. Valid operators are Exists and Equal. Defaults to Equal. Exists is equivalent to wildcard for value, so that a pod can tolerate all taints of a particular category. + type: string + tolerationSeconds: + description: TolerationSeconds represents the period of time the toleration (which must be of effect NoExecute, otherwise this field is ignored) tolerates the taint. By default, it is not set, which means tolerate the taint forever (do not evict). Zero and negative values will be treated as 0 (evict immediately) by the system. + type: integer + format: int64 + value: + description: Value is the taint value the toleration matches to. If the operator is Exists, the value should be empty, otherwise just a regular string. + type: string + serviceType: + description: Optional service type for Kubernetes solver service. Supported values are NodePort or ClusterIP. If unset, defaults to NodePort. + type: string + selector: + description: Selector selects a set of DNSNames on the Certificate resource that should be solved using this challenge solver. If not specified, the solver will be treated as the 'default' solver with the lowest priority, i.e. if any other solver has a more specific match, it will be used instead. + type: object + properties: + dnsNames: + description: List of DNSNames that this solver will be used to solve. If specified and a match is found, a dnsNames selector will take precedence over a dnsZones selector. If multiple solvers match with the same dnsNames value, the solver with the most matching labels in matchLabels will be selected. If neither has more matches, the solver defined earlier in the list will be selected. + type: array + items: + type: string + dnsZones: + description: List of DNSZones that this solver will be used to solve. The most specific DNS zone match specified here will take precedence over other DNS zone matches, so a solver specifying sys.example.com will be selected over one specifying example.com for the domain www.sys.example.com. If multiple solvers match with the same dnsZones value, the solver with the most matching labels in matchLabels will be selected. If neither has more matches, the solver defined earlier in the list will be selected. + type: array + items: + type: string + matchLabels: + description: A label selector that is used to refine the set of certificate's that this challenge solver will apply to. + type: object + additionalProperties: + type: string + ca: + description: CA configures this issuer to sign certificates using a signing CA keypair stored in a Secret resource. This is used to build internal PKIs that are managed by cert-manager. + type: object + required: + - secretName + properties: + crlDistributionPoints: + description: The CRL distribution points is an X.509 v3 certificate extension which identifies the location of the CRL from which the revocation of this certificate can be checked. If not set, certificates will be issued without distribution points set. + type: array + items: + type: string + ocspServers: + description: The OCSP server list is an X.509 v3 extension that defines a list of URLs of OCSP responders. The OCSP responders can be queried for the revocation status of an issued certificate. If not set, the certificate will be issued with no OCSP servers set. For example, an OCSP server URL could be "http://ocsp.int-x3.letsencrypt.org". + type: array + items: + type: string + secretName: + description: SecretName is the name of the secret used to sign Certificates issued by this Issuer. + type: string + selfSigned: + description: SelfSigned configures this issuer to 'self sign' certificates using the private key used to create the CertificateRequest object. + type: object + properties: + crlDistributionPoints: + description: The CRL distribution points is an X.509 v3 certificate extension which identifies the location of the CRL from which the revocation of this certificate can be checked. If not set certificate will be issued without CDP. Values are strings. + type: array + items: + type: string + vault: + description: Vault configures this issuer to sign certificates using a HashiCorp Vault PKI backend. + type: object + required: + - auth + - path + - server + properties: + auth: + description: Auth configures how cert-manager authenticates with the Vault server. + type: object + properties: + appRole: + description: AppRole authenticates with Vault using the App Role auth mechanism, with the role and secret stored in a Kubernetes Secret resource. + type: object + required: + - path + - roleId + - secretRef + properties: + path: + description: 'Path where the App Role authentication backend is mounted in Vault, e.g: "approle"' + type: string + roleId: + description: RoleID configured in the App Role authentication backend when setting up the authentication backend in Vault. + type: string + secretRef: + description: Reference to a key in a Secret that contains the App Role secret used to authenticate with Vault. The `key` field must be specified and denotes which entry within the Secret resource is used as the app role secret. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + kubernetes: + description: Kubernetes authenticates with Vault by passing the ServiceAccount token stored in the named Secret resource to the Vault server. + type: object + required: + - role + - secretRef + properties: + mountPath: + description: The Vault mountPath here is the mount path to use when authenticating with Vault. For example, setting a value to `/v1/auth/foo`, will use the path `/v1/auth/foo/login` to authenticate with Vault. If unspecified, the default value "/v1/auth/kubernetes" will be used. + type: string + role: + description: A required field containing the Vault Role to assume. A Role binds a Kubernetes ServiceAccount with a set of Vault policies. + type: string + secretRef: + description: The required Secret field containing a Kubernetes ServiceAccount JWT used for authenticating with Vault. Use of 'ambient credentials' is not supported. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + tokenSecretRef: + description: TokenSecretRef authenticates with Vault by presenting a token. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + caBundle: + description: Base64-encoded bundle of PEM CAs which will be used to validate the certificate chain presented by Vault. Only used if using HTTPS to connect to Vault and ignored for HTTP connections. Mutually exclusive with CABundleSecretRef. If neither CABundle nor CABundleSecretRef are defined, the certificate bundle in the cert-manager controller container is used to validate the TLS connection. + type: string + format: byte + caBundleSecretRef: + description: Reference to a Secret containing a bundle of PEM-encoded CAs to use when verifying the certificate chain presented by Vault when using HTTPS. Mutually exclusive with CABundle. If neither CABundle nor CABundleSecretRef are defined, the certificate bundle in the cert-manager controller container is used to validate the TLS connection. If no key for the Secret is specified, cert-manager will default to 'ca.crt'. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + namespace: + description: 'Name of the vault namespace. Namespaces is a set of features within Vault Enterprise that allows Vault environments to support Secure Multi-tenancy. e.g: "ns1" More about namespaces can be found here https://www.vaultproject.io/docs/enterprise/namespaces' + type: string + path: + description: 'Path is the mount path of the Vault PKI backend''s `sign` endpoint, e.g: "my_pki_mount/sign/my-role-name".' + type: string + server: + description: 'Server is the connection address for the Vault server, e.g: "https://vault.example.com:8200".' + type: string + venafi: + description: Venafi configures this issuer to sign certificates using a Venafi TPP or Venafi Cloud policy zone. + type: object + required: + - zone + properties: + cloud: + description: Cloud specifies the Venafi cloud configuration settings. Only one of TPP or Cloud may be specified. + type: object + required: + - apiTokenSecretRef + properties: + apiTokenSecretRef: + description: APITokenSecretRef is a secret key selector for the Venafi Cloud API token. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + url: + description: URL is the base URL for Venafi Cloud. Defaults to "https://api.venafi.cloud/v1". + type: string + tpp: + description: TPP specifies Trust Protection Platform configuration settings. Only one of TPP or Cloud may be specified. + type: object + required: + - credentialsRef + - url + properties: + caBundle: + description: Base64-encoded bundle of PEM CAs which will be used to validate the certificate chain presented by the TPP server. Only used if using HTTPS; ignored for HTTP. If undefined, the certificate bundle in the cert-manager controller container is used to validate the chain. + type: string + format: byte + credentialsRef: + description: CredentialsRef is a reference to a Secret containing the username and password for the TPP server. The secret must contain two keys, 'username' and 'password'. + type: object + required: + - name + properties: + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + url: + description: 'URL is the base URL for the vedsdk endpoint of the Venafi TPP instance, for example: "https://tpp.example.com/vedsdk".' + type: string + zone: + description: Zone is the Venafi Policy Zone to use for this issuer. All requests made to the Venafi platform will be restricted by the named zone policy. This field is required. + type: string + status: + description: Status of the Issuer. This is set and managed automatically. + type: object + properties: + acme: + description: ACME specific status options. This field should only be set if the Issuer is configured to use an ACME server to issue certificates. + type: object + properties: + lastRegisteredEmail: + description: LastRegisteredEmail is the email associated with the latest registered ACME account, in order to track changes made to registered account associated with the Issuer + type: string + uri: + description: URI is the unique account identifier, which can also be used to retrieve account details from the CA + type: string + conditions: + description: List of status conditions to indicate the status of a CertificateRequest. Known condition types are `Ready`. + type: array + items: + description: IssuerCondition contains condition information for an Issuer. + type: object + required: + - status + - type + properties: + lastTransitionTime: + description: LastTransitionTime is the timestamp corresponding to the last status change of this condition. + type: string + format: date-time + message: + description: Message is a human readable description of the details of the last transition, complementing reason. + type: string + observedGeneration: + description: If set, this represents the .metadata.generation that the condition was set based upon. For instance, if .metadata.generation is currently 12, but the .status.condition[x].observedGeneration is 9, the condition is out of date with respect to the current state of the Issuer. + type: integer + format: int64 + reason: + description: Reason is a brief machine readable explanation for the condition's last transition. + type: string + status: + description: Status of the condition, one of (`True`, `False`, `Unknown`). + type: string + enum: + - "True" + - "False" + - Unknown + type: + description: Type of the condition, known values are (`Ready`). + type: string + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + served: true + storage: true +--- +# Source: cert-manager/deploy/crds/crd-certificates.yaml +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: certificates.cert-manager.io + labels: + app: 'cert-manager' + app.kubernetes.io/name: 'cert-manager' + app.kubernetes.io/instance: 'cert-manager' + # Generated labels + app.kubernetes.io/version: "{{ cert_manager_version }}" +spec: + group: cert-manager.io + names: + kind: Certificate + listKind: CertificateList + plural: certificates + shortNames: + - cert + - certs + singular: certificate + categories: + - cert-manager + scope: Namespaced + versions: + - name: v1 + subresources: + status: {} + additionalPrinterColumns: + - jsonPath: .status.conditions[?(@.type=="Ready")].status + name: Ready + type: string + - jsonPath: .spec.secretName + name: Secret + type: string + - jsonPath: .spec.issuerRef.name + name: Issuer + priority: 1 + type: string + - jsonPath: .status.conditions[?(@.type=="Ready")].message + name: Status + priority: 1 + type: string + - jsonPath: .metadata.creationTimestamp + description: CreationTimestamp is a timestamp representing the server time when this object was created. It is not guaranteed to be set in happens-before order across separate operations. Clients may not set this value. It is represented in RFC3339 form and is in UTC. + name: Age + type: date + schema: + openAPIV3Schema: + description: "A Certificate resource should be created to ensure an up to date and signed x509 certificate is stored in the Kubernetes Secret resource named in `spec.secretName`. \n The stored certificate will be renewed before it expires (as configured by `spec.renewBefore`)." + type: object + required: + - spec + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation of an object. Servers should convert recognized schemas to the latest internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this object represents. Servers may infer this from the endpoint the client submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: Desired state of the Certificate resource. + type: object + required: + - issuerRef + - secretName + properties: + additionalOutputFormats: + description: AdditionalOutputFormats defines extra output formats of the private key and signed certificate chain to be written to this Certificate's target Secret. This is an Alpha Feature and is only enabled with the `--feature-gates=AdditionalCertificateOutputFormats=true` option on both the controller and webhook components. + type: array + items: + description: CertificateAdditionalOutputFormat defines an additional output format of a Certificate resource. These contain supplementary data formats of the signed certificate chain and paired private key. + type: object + required: + - type + properties: + type: + description: Type is the name of the format type that should be written to the Certificate's target Secret. + type: string + enum: + - DER + - CombinedPEM + commonName: + description: 'CommonName is a common name to be used on the Certificate. The CommonName should have a length of 64 characters or fewer to avoid generating invalid CSRs. This value is ignored by TLS clients when any subject alt name is set. This is x509 behaviour: https://tools.ietf.org/html/rfc6125#section-6.4.4' + type: string + dnsNames: + description: DNSNames is a list of DNS subjectAltNames to be set on the Certificate. + type: array + items: + type: string + duration: + description: The requested 'duration' (i.e. lifetime) of the Certificate. This option may be ignored/overridden by some issuer types. If unset this defaults to 90 days. Certificate will be renewed either 2/3 through its duration or `renewBefore` period before its expiry, whichever is later. Minimum accepted duration is 1 hour. Value must be in units accepted by Go time.ParseDuration https://golang.org/pkg/time/#ParseDuration + type: string + emailAddresses: + description: EmailAddresses is a list of email subjectAltNames to be set on the Certificate. + type: array + items: + type: string + encodeUsagesInRequest: + description: EncodeUsagesInRequest controls whether key usages should be present in the CertificateRequest + type: boolean + ipAddresses: + description: IPAddresses is a list of IP address subjectAltNames to be set on the Certificate. + type: array + items: + type: string + isCA: + description: IsCA will mark this Certificate as valid for certificate signing. This will automatically add the `cert sign` usage to the list of `usages`. + type: boolean + issuerRef: + description: IssuerRef is a reference to the issuer for this certificate. If the `kind` field is not set, or set to `Issuer`, an Issuer resource with the given name in the same namespace as the Certificate will be used. If the `kind` field is set to `ClusterIssuer`, a ClusterIssuer with the provided name will be used. The `name` field in this stanza is required at all times. + type: object + required: + - name + properties: + group: + description: Group of the resource being referred to. + type: string + kind: + description: Kind of the resource being referred to. + type: string + name: + description: Name of the resource being referred to. + type: string + keystores: + description: Keystores configures additional keystore output formats stored in the `secretName` Secret resource. + type: object + properties: + jks: + description: JKS configures options for storing a JKS keystore in the `spec.secretName` Secret resource. + type: object + required: + - create + - passwordSecretRef + properties: + create: + description: Create enables JKS keystore creation for the Certificate. If true, a file named `keystore.jks` will be created in the target Secret resource, encrypted using the password stored in `passwordSecretRef`. The keystore file will be updated immediately. A file named `truststore.jks` will also be created in the target Secret resource, encrypted using the password stored in `passwordSecretRef` containing the issuing Certificate Authority + type: boolean + passwordSecretRef: + description: PasswordSecretRef is a reference to a key in a Secret resource containing the password used to encrypt the JKS keystore. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + pkcs12: + description: PKCS12 configures options for storing a PKCS12 keystore in the `spec.secretName` Secret resource. + type: object + required: + - create + - passwordSecretRef + properties: + create: + description: Create enables PKCS12 keystore creation for the Certificate. If true, a file named `keystore.p12` will be created in the target Secret resource, encrypted using the password stored in `passwordSecretRef`. The keystore file will be updated immediately. A file named `truststore.p12` will also be created in the target Secret resource, encrypted using the password stored in `passwordSecretRef` containing the issuing Certificate Authority + type: boolean + passwordSecretRef: + description: PasswordSecretRef is a reference to a key in a Secret resource containing the password used to encrypt the PKCS12 keystore. + type: object + required: + - name + properties: + key: + description: The key of the entry in the Secret resource's `data` field to be used. Some instances of this field may be defaulted, in others it may be required. + type: string + name: + description: 'Name of the resource being referred to. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + literalSubject: + description: LiteralSubject is an LDAP formatted string that represents the [X.509 Subject field](https://datatracker.ietf.org/doc/html/rfc5280#section-4.1.2.6). Use this *instead* of the Subject field if you need to ensure the correct ordering of the RDN sequence, such as when issuing certs for LDAP authentication. See https://github.com/cert-manager/cert-manager/issues/3203, https://github.com/cert-manager/cert-manager/issues/4424. This field is alpha level and is only supported by cert-manager installations where LiteralCertificateSubject feature gate is enabled on both cert-manager controller and webhook. + type: string + privateKey: + description: Options to control private keys used for the Certificate. + type: object + properties: + algorithm: + description: Algorithm is the private key algorithm of the corresponding private key for this certificate. If provided, allowed values are either `RSA`,`Ed25519` or `ECDSA` If `algorithm` is specified and `size` is not provided, key size of 256 will be used for `ECDSA` key algorithm and key size of 2048 will be used for `RSA` key algorithm. key size is ignored when using the `Ed25519` key algorithm. + type: string + enum: + - RSA + - ECDSA + - Ed25519 + encoding: + description: The private key cryptography standards (PKCS) encoding for this certificate's private key to be encoded in. If provided, allowed values are `PKCS1` and `PKCS8` standing for PKCS#1 and PKCS#8, respectively. Defaults to `PKCS1` if not specified. + type: string + enum: + - PKCS1 + - PKCS8 + rotationPolicy: + description: RotationPolicy controls how private keys should be regenerated when a re-issuance is being processed. If set to Never, a private key will only be generated if one does not already exist in the target `spec.secretName`. If one does exists but it does not have the correct algorithm or size, a warning will be raised to await user intervention. If set to Always, a private key matching the specified requirements will be generated whenever a re-issuance occurs. Default is 'Never' for backward compatibility. + type: string + enum: + - Never + - Always + size: + description: Size is the key bit size of the corresponding private key for this certificate. If `algorithm` is set to `RSA`, valid values are `2048`, `4096` or `8192`, and will default to `2048` if not specified. If `algorithm` is set to `ECDSA`, valid values are `256`, `384` or `521`, and will default to `256` if not specified. If `algorithm` is set to `Ed25519`, Size is ignored. No other values are allowed. + type: integer + renewBefore: + description: How long before the currently issued certificate's expiry cert-manager should renew the certificate. The default is 2/3 of the issued certificate's duration. Minimum accepted value is 5 minutes. Value must be in units accepted by Go time.ParseDuration https://golang.org/pkg/time/#ParseDuration + type: string + revisionHistoryLimit: + description: revisionHistoryLimit is the maximum number of CertificateRequest revisions that are maintained in the Certificate's history. Each revision represents a single `CertificateRequest` created by this Certificate, either when it was created, renewed, or Spec was changed. Revisions will be removed by oldest first if the number of revisions exceeds this number. If set, revisionHistoryLimit must be a value of `1` or greater. If unset (`nil`), revisions will not be garbage collected. Default value is `nil`. + type: integer + format: int32 + secretName: + description: SecretName is the name of the secret resource that will be automatically created and managed by this Certificate resource. It will be populated with a private key and certificate, signed by the denoted issuer. + type: string + secretTemplate: + description: SecretTemplate defines annotations and labels to be copied to the Certificate's Secret. Labels and annotations on the Secret will be changed as they appear on the SecretTemplate when added or removed. SecretTemplate annotations are added in conjunction with, and cannot overwrite, the base set of annotations cert-manager sets on the Certificate's Secret. + type: object + properties: + annotations: + description: Annotations is a key value map to be copied to the target Kubernetes Secret. + type: object + additionalProperties: + type: string + labels: + description: Labels is a key value map to be copied to the target Kubernetes Secret. + type: object + additionalProperties: + type: string + subject: + description: Full X509 name specification (https://golang.org/pkg/crypto/x509/pkix/#Name). + type: object + properties: + countries: + description: Countries to be used on the Certificate. + type: array + items: + type: string + localities: + description: Cities to be used on the Certificate. + type: array + items: + type: string + organizationalUnits: + description: Organizational Units to be used on the Certificate. + type: array + items: + type: string + organizations: + description: Organizations to be used on the Certificate. + type: array + items: + type: string + postalCodes: + description: Postal codes to be used on the Certificate. + type: array + items: + type: string + provinces: + description: State/Provinces to be used on the Certificate. + type: array + items: + type: string + serialNumber: + description: Serial number to be used on the Certificate. + type: string + streetAddresses: + description: Street addresses to be used on the Certificate. + type: array + items: + type: string + uris: + description: URIs is a list of URI subjectAltNames to be set on the Certificate. + type: array + items: + type: string + usages: + description: Usages is the set of x509 usages that are requested for the certificate. Defaults to `digital signature` and `key encipherment` if not specified. + type: array + items: + description: "KeyUsage specifies valid usage contexts for keys. See: https://tools.ietf.org/html/rfc5280#section-4.2.1.3 https://tools.ietf.org/html/rfc5280#section-4.2.1.12 \n Valid KeyUsage values are as follows: \"signing\", \"digital signature\", \"content commitment\", \"key encipherment\", \"key agreement\", \"data encipherment\", \"cert sign\", \"crl sign\", \"encipher only\", \"decipher only\", \"any\", \"server auth\", \"client auth\", \"code signing\", \"email protection\", \"s/mime\", \"ipsec end system\", \"ipsec tunnel\", \"ipsec user\", \"timestamping\", \"ocsp signing\", \"microsoft sgc\", \"netscape sgc\"" + type: string + enum: + - signing + - digital signature + - content commitment + - key encipherment + - key agreement + - data encipherment + - cert sign + - crl sign + - encipher only + - decipher only + - any + - server auth + - client auth + - code signing + - email protection + - s/mime + - ipsec end system + - ipsec tunnel + - ipsec user + - timestamping + - ocsp signing + - microsoft sgc + - netscape sgc + status: + description: Status of the Certificate. This is set and managed automatically. + type: object + properties: + conditions: + description: List of status conditions to indicate the status of certificates. Known condition types are `Ready` and `Issuing`. + type: array + items: + description: CertificateCondition contains condition information for an Certificate. + type: object + required: + - status + - type + properties: + lastTransitionTime: + description: LastTransitionTime is the timestamp corresponding to the last status change of this condition. + type: string + format: date-time + message: + description: Message is a human readable description of the details of the last transition, complementing reason. + type: string + observedGeneration: + description: If set, this represents the .metadata.generation that the condition was set based upon. For instance, if .metadata.generation is currently 12, but the .status.condition[x].observedGeneration is 9, the condition is out of date with respect to the current state of the Certificate. + type: integer + format: int64 + reason: + description: Reason is a brief machine readable explanation for the condition's last transition. + type: string + status: + description: Status of the condition, one of (`True`, `False`, `Unknown`). + type: string + enum: + - "True" + - "False" + - Unknown + type: + description: Type of the condition, known values are (`Ready`, `Issuing`). + type: string + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + failedIssuanceAttempts: + description: The number of continuous failed issuance attempts up till now. This field gets removed (if set) on a successful issuance and gets set to 1 if unset and an issuance has failed. If an issuance has failed, the delay till the next issuance will be calculated using formula time.Hour * 2 ^ (failedIssuanceAttempts - 1). + type: integer + lastFailureTime: + description: LastFailureTime is the time as recorded by the Certificate controller of the most recent failure to complete a CertificateRequest for this Certificate resource. If set, cert-manager will not re-request another Certificate until 1 hour has elapsed from this time. + type: string + format: date-time + nextPrivateKeySecretName: + description: The name of the Secret resource containing the private key to be used for the next certificate iteration. The keymanager controller will automatically set this field if the `Issuing` condition is set to `True`. It will automatically unset this field when the Issuing condition is not set or False. + type: string + notAfter: + description: The expiration time of the certificate stored in the secret named by this resource in `spec.secretName`. + type: string + format: date-time + notBefore: + description: The time after which the certificate stored in the secret named by this resource in spec.secretName is valid. + type: string + format: date-time + renewalTime: + description: RenewalTime is the time at which the certificate will be next renewed. If not set, no upcoming renewal is scheduled. + type: string + format: date-time + revision: + description: "The current 'revision' of the certificate as issued. \n When a CertificateRequest resource is created, it will have the `cert-manager.io/certificate-revision` set to one greater than the current value of this field. \n Upon issuance, this field will be set to the value of the annotation on the CertificateRequest resource used to issue the certificate. \n Persisting the value on the CertificateRequest resource allows the certificates controller to know whether a request is part of an old issuance or if it is part of the ongoing revision's issuance by checking if the revision value in the annotation is greater than this field." + type: integer + served: true + storage: true +--- +# Source: cert-manager/deploy/crds/crd-orders.yaml +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: orders.acme.cert-manager.io + labels: + app: 'cert-manager' + app.kubernetes.io/name: 'cert-manager' + app.kubernetes.io/instance: 'cert-manager' + # Generated labels + app.kubernetes.io/version: "{{ cert_manager_version }}" +spec: + group: acme.cert-manager.io + names: + kind: Order + listKind: OrderList + plural: orders + singular: order + categories: + - cert-manager + - cert-manager-acme + scope: Namespaced + versions: + - name: v1 + subresources: + status: {} + additionalPrinterColumns: + - jsonPath: .status.state + name: State + type: string + - jsonPath: .spec.issuerRef.name + name: Issuer + priority: 1 + type: string + - jsonPath: .status.reason + name: Reason + priority: 1 + type: string + - jsonPath: .metadata.creationTimestamp + description: CreationTimestamp is a timestamp representing the server time when this object was created. It is not guaranteed to be set in happens-before order across separate operations. Clients may not set this value. It is represented in RFC3339 form and is in UTC. + name: Age + type: date + schema: + openAPIV3Schema: + description: Order is a type to represent an Order with an ACME server + type: object + required: + - metadata + - spec + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation of an object. Servers should convert recognized schemas to the latest internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this object represents. Servers may infer this from the endpoint the client submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + type: object + required: + - issuerRef + - request + properties: + commonName: + description: CommonName is the common name as specified on the DER encoded CSR. If specified, this value must also be present in `dnsNames` or `ipAddresses`. This field must match the corresponding field on the DER encoded CSR. + type: string + dnsNames: + description: DNSNames is a list of DNS names that should be included as part of the Order validation process. This field must match the corresponding field on the DER encoded CSR. + type: array + items: + type: string + duration: + description: Duration is the duration for the not after date for the requested certificate. this is set on order creation as pe the ACME spec. + type: string + ipAddresses: + description: IPAddresses is a list of IP addresses that should be included as part of the Order validation process. This field must match the corresponding field on the DER encoded CSR. + type: array + items: + type: string + issuerRef: + description: IssuerRef references a properly configured ACME-type Issuer which should be used to create this Order. If the Issuer does not exist, processing will be retried. If the Issuer is not an 'ACME' Issuer, an error will be returned and the Order will be marked as failed. + type: object + required: + - name + properties: + group: + description: Group of the resource being referred to. + type: string + kind: + description: Kind of the resource being referred to. + type: string + name: + description: Name of the resource being referred to. + type: string + request: + description: Certificate signing request bytes in DER encoding. This will be used when finalizing the order. This field must be set on the order. + type: string + format: byte + status: + type: object + properties: + authorizations: + description: Authorizations contains data returned from the ACME server on what authorizations must be completed in order to validate the DNS names specified on the Order. + type: array + items: + description: ACMEAuthorization contains data returned from the ACME server on an authorization that must be completed in order validate a DNS name on an ACME Order resource. + type: object + required: + - url + properties: + challenges: + description: Challenges specifies the challenge types offered by the ACME server. One of these challenge types will be selected when validating the DNS name and an appropriate Challenge resource will be created to perform the ACME challenge process. + type: array + items: + description: Challenge specifies a challenge offered by the ACME server for an Order. An appropriate Challenge resource can be created to perform the ACME challenge process. + type: object + required: + - token + - type + - url + properties: + token: + description: Token is the token that must be presented for this challenge. This is used to compute the 'key' that must also be presented. + type: string + type: + description: Type is the type of challenge being offered, e.g. 'http-01', 'dns-01', 'tls-sni-01', etc. This is the raw value retrieved from the ACME server. Only 'http-01' and 'dns-01' are supported by cert-manager, other values will be ignored. + type: string + url: + description: URL is the URL of this challenge. It can be used to retrieve additional metadata about the Challenge from the ACME server. + type: string + identifier: + description: Identifier is the DNS name to be validated as part of this authorization + type: string + initialState: + description: InitialState is the initial state of the ACME authorization when first fetched from the ACME server. If an Authorization is already 'valid', the Order controller will not create a Challenge resource for the authorization. This will occur when working with an ACME server that enables 'authz reuse' (such as Let's Encrypt's production endpoint). If not set and 'identifier' is set, the state is assumed to be pending and a Challenge will be created. + type: string + enum: + - valid + - ready + - pending + - processing + - invalid + - expired + - errored + url: + description: URL is the URL of the Authorization that must be completed + type: string + wildcard: + description: Wildcard will be true if this authorization is for a wildcard DNS name. If this is true, the identifier will be the *non-wildcard* version of the DNS name. For example, if '*.example.com' is the DNS name being validated, this field will be 'true' and the 'identifier' field will be 'example.com'. + type: boolean + certificate: + description: Certificate is a copy of the PEM encoded certificate for this Order. This field will be populated after the order has been successfully finalized with the ACME server, and the order has transitioned to the 'valid' state. + type: string + format: byte + failureTime: + description: FailureTime stores the time that this order failed. This is used to influence garbage collection and back-off. + type: string + format: date-time + finalizeURL: + description: FinalizeURL of the Order. This is used to obtain certificates for this order once it has been completed. + type: string + reason: + description: Reason optionally provides more information about a why the order is in the current state. + type: string + state: + description: State contains the current state of this Order resource. States 'success' and 'expired' are 'final' + type: string + enum: + - valid + - ready + - pending + - processing + - invalid + - expired + - errored + url: + description: URL of the Order. This will initially be empty when the resource is first created. The Order controller will populate this field when the Order is first processed. This field will be immutable after it is initially set. + type: string + served: true + storage: true diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/templates/cert-manager.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/templates/cert-manager.yml.j2 new file mode 100644 index 0000000..d4e791c --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/cert_manager/templates/cert-manager.yml.j2 @@ -0,0 +1,1224 @@ +# Copyright 2022 The cert-manager Authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +--- +apiVersion: v1 +kind: Namespace +metadata: + name: {{ cert_manager_namespace }} +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/cainjector-serviceaccount.yaml +apiVersion: v1 +kind: ServiceAccount +automountServiceAccountToken: true +metadata: + name: cert-manager-cainjector + namespace: {{ cert_manager_namespace }} + labels: + app: cainjector + app.kubernetes.io/name: cainjector + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "cainjector" + app.kubernetes.io/version: "{{ cert_manager_version }}" +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/serviceaccount.yaml +apiVersion: v1 +kind: ServiceAccount +automountServiceAccountToken: true +metadata: + name: cert-manager + namespace: {{ cert_manager_namespace }} + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/webhook-serviceaccount.yaml +apiVersion: v1 +kind: ServiceAccount +automountServiceAccountToken: true +metadata: + name: cert-manager-webhook + namespace: {{ cert_manager_namespace }} + labels: + app: webhook + app.kubernetes.io/name: webhook + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "webhook" + app.kubernetes.io/version: "{{ cert_manager_version }}" +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/webhook-config.yaml +apiVersion: v1 +kind: ConfigMap +metadata: + name: cert-manager-webhook + namespace: {{ cert_manager_namespace }} + labels: + app: webhook + app.kubernetes.io/name: webhook + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "webhook" +data: +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/cainjector-rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cert-manager-cainjector + labels: + app: cainjector + app.kubernetes.io/name: cainjector + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "cainjector" + app.kubernetes.io/version: "{{ cert_manager_version }}" +rules: + - apiGroups: ["cert-manager.io"] + resources: ["certificates"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["get", "create", "update", "patch"] + - apiGroups: ["admissionregistration.k8s.io"] + resources: ["validatingwebhookconfigurations", "mutatingwebhookconfigurations"] + verbs: ["get", "list", "watch", "update"] + - apiGroups: ["apiregistration.k8s.io"] + resources: ["apiservices"] + verbs: ["get", "list", "watch", "update"] + - apiGroups: ["apiextensions.k8s.io"] + resources: ["customresourcedefinitions"] + verbs: ["get", "list", "watch", "update"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +# Issuer controller role +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cert-manager-controller-issuers + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +rules: + - apiGroups: ["cert-manager.io"] + resources: ["issuers", "issuers/status"] + verbs: ["update", "patch"] + - apiGroups: ["cert-manager.io"] + resources: ["issuers"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list", "watch", "create", "update", "delete"] + - apiGroups: [""] + resources: ["events"] + verbs: ["create", "patch"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +# ClusterIssuer controller role +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cert-manager-controller-clusterissuers + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +rules: + - apiGroups: ["cert-manager.io"] + resources: ["clusterissuers", "clusterissuers/status"] + verbs: ["update", "patch"] + - apiGroups: ["cert-manager.io"] + resources: ["clusterissuers"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list", "watch", "create", "update", "delete"] + - apiGroups: [""] + resources: ["events"] + verbs: ["create", "patch"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +# Certificates controller role +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cert-manager-controller-certificates + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +rules: + - apiGroups: ["cert-manager.io"] + resources: ["certificates", "certificates/status", "certificaterequests", "certificaterequests/status"] + verbs: ["update", "patch"] + - apiGroups: ["cert-manager.io"] + resources: ["certificates", "certificaterequests", "clusterissuers", "issuers"] + verbs: ["get", "list", "watch"] + # We require these rules to support users with the OwnerReferencesPermissionEnforcement + # admission controller enabled: + # https://kubernetes.io/docs/reference/access-authn-authz/admission-controllers/#ownerreferencespermissionenforcement + - apiGroups: ["cert-manager.io"] + resources: ["certificates/finalizers", "certificaterequests/finalizers"] + verbs: ["update"] + - apiGroups: ["acme.cert-manager.io"] + resources: ["orders"] + verbs: ["create", "delete", "get", "list", "watch"] + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list", "watch", "create", "update", "delete", "patch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["create", "patch"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +# Orders controller role +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cert-manager-controller-orders + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +rules: + - apiGroups: ["acme.cert-manager.io"] + resources: ["orders", "orders/status"] + verbs: ["update", "patch"] + - apiGroups: ["acme.cert-manager.io"] + resources: ["orders", "challenges"] + verbs: ["get", "list", "watch"] + - apiGroups: ["cert-manager.io"] + resources: ["clusterissuers", "issuers"] + verbs: ["get", "list", "watch"] + - apiGroups: ["acme.cert-manager.io"] + resources: ["challenges"] + verbs: ["create", "delete"] + # We require these rules to support users with the OwnerReferencesPermissionEnforcement + # admission controller enabled: + # https://kubernetes.io/docs/reference/access-authn-authz/admission-controllers/#ownerreferencespermissionenforcement + - apiGroups: ["acme.cert-manager.io"] + resources: ["orders/finalizers"] + verbs: ["update"] + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["create", "patch"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +# Challenges controller role +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cert-manager-controller-challenges + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +rules: + # Use to update challenge resource status + - apiGroups: ["acme.cert-manager.io"] + resources: ["challenges", "challenges/status"] + verbs: ["update", "patch"] + # Used to watch challenge resources + - apiGroups: ["acme.cert-manager.io"] + resources: ["challenges"] + verbs: ["get", "list", "watch"] + # Used to watch challenges, issuer and clusterissuer resources + - apiGroups: ["cert-manager.io"] + resources: ["issuers", "clusterissuers"] + verbs: ["get", "list", "watch"] + # Need to be able to retrieve ACME account private key to complete challenges + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list", "watch"] + # Used to create events + - apiGroups: [""] + resources: ["events"] + verbs: ["create", "patch"] + # HTTP01 rules + - apiGroups: [""] + resources: ["pods", "services"] + verbs: ["get", "list", "watch", "create", "delete"] + - apiGroups: ["networking.k8s.io"] + resources: ["ingresses"] + verbs: ["get", "list", "watch", "create", "delete", "update"] + - apiGroups: [ "gateway.networking.k8s.io" ] + resources: [ "httproutes" ] + verbs: ["get", "list", "watch", "create", "delete", "update"] + # We require the ability to specify a custom hostname when we are creating + # new ingress resources. + # See: https://github.com/openshift/origin/blob/21f191775636f9acadb44fa42beeb4f75b255532/pkg/route/apiserver/admission/ingress_admission.go#L84-L148 + - apiGroups: ["route.openshift.io"] + resources: ["routes/custom-host"] + verbs: ["create"] + # We require these rules to support users with the OwnerReferencesPermissionEnforcement + # admission controller enabled: + # https://kubernetes.io/docs/reference/access-authn-authz/admission-controllers/#ownerreferencespermissionenforcement + - apiGroups: ["acme.cert-manager.io"] + resources: ["challenges/finalizers"] + verbs: ["update"] + # DNS01 rules (duplicated above) + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list", "watch"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +# ingress-shim controller role +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cert-manager-controller-ingress-shim + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +rules: + - apiGroups: ["cert-manager.io"] + resources: ["certificates", "certificaterequests"] + verbs: ["create", "update", "delete"] + - apiGroups: ["cert-manager.io"] + resources: ["certificates", "certificaterequests", "issuers", "clusterissuers"] + verbs: ["get", "list", "watch"] + - apiGroups: ["networking.k8s.io"] + resources: ["ingresses"] + verbs: ["get", "list", "watch"] + # We require these rules to support users with the OwnerReferencesPermissionEnforcement + # admission controller enabled: + # https://kubernetes.io/docs/reference/access-authn-authz/admission-controllers/#ownerreferencespermissionenforcement + - apiGroups: ["networking.k8s.io"] + resources: ["ingresses/finalizers"] + verbs: ["update"] + - apiGroups: ["gateway.networking.k8s.io"] + resources: ["gateways", "httproutes"] + verbs: ["get", "list", "watch"] + - apiGroups: ["gateway.networking.k8s.io"] + resources: ["gateways/finalizers", "httproutes/finalizers"] + verbs: ["update"] + - apiGroups: [""] + resources: ["events"] + verbs: ["create", "patch"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cert-manager-view + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" + rbac.authorization.k8s.io/aggregate-to-view: "true" + rbac.authorization.k8s.io/aggregate-to-edit: "true" + rbac.authorization.k8s.io/aggregate-to-admin: "true" +rules: + - apiGroups: ["cert-manager.io"] + resources: ["certificates", "certificaterequests", "issuers"] + verbs: ["get", "list", "watch"] + - apiGroups: ["acme.cert-manager.io"] + resources: ["challenges", "orders"] + verbs: ["get", "list", "watch"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cert-manager-edit + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" + rbac.authorization.k8s.io/aggregate-to-edit: "true" + rbac.authorization.k8s.io/aggregate-to-admin: "true" +rules: + - apiGroups: ["cert-manager.io"] + resources: ["certificates", "certificaterequests", "issuers"] + verbs: ["create", "delete", "deletecollection", "patch", "update"] + - apiGroups: ["cert-manager.io"] + resources: ["certificates/status"] + verbs: ["update"] + - apiGroups: ["acme.cert-manager.io"] + resources: ["challenges", "orders"] + verbs: ["create", "delete", "deletecollection", "patch", "update"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +# Permission to approve CertificateRequests referencing cert-manager.io Issuers and ClusterIssuers +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cert-manager-controller-approve:cert-manager-io + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "cert-manager" + app.kubernetes.io/version: "{{ cert_manager_version }}" +rules: + - apiGroups: ["cert-manager.io"] + resources: ["signers"] + verbs: ["approve"] + resourceNames: ["issuers.cert-manager.io/*", "clusterissuers.cert-manager.io/*"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +# Permission to: +# - Update and sign CertificatSigningeRequests referencing cert-manager.io Issuers and ClusterIssuers +# - Perform SubjectAccessReviews to test whether users are able to reference Namespaced Issuers +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cert-manager-controller-certificatesigningrequests + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "cert-manager" + app.kubernetes.io/version: "{{ cert_manager_version }}" +rules: + - apiGroups: ["certificates.k8s.io"] + resources: ["certificatesigningrequests"] + verbs: ["get", "list", "watch", "update"] + - apiGroups: ["certificates.k8s.io"] + resources: ["certificatesigningrequests/status"] + verbs: ["update", "patch"] + - apiGroups: ["certificates.k8s.io"] + resources: ["signers"] + resourceNames: ["issuers.cert-manager.io/*", "clusterissuers.cert-manager.io/*"] + verbs: ["sign"] + - apiGroups: ["authorization.k8s.io"] + resources: ["subjectaccessreviews"] + verbs: ["create"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/webhook-rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cert-manager-webhook:subjectaccessreviews + labels: + app: webhook + app.kubernetes.io/name: webhook + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "webhook" + app.kubernetes.io/version: "{{ cert_manager_version }}" +rules: +- apiGroups: ["authorization.k8s.io"] + resources: ["subjectaccessreviews"] + verbs: ["create"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/cainjector-rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cert-manager-cainjector + labels: + app: cainjector + app.kubernetes.io/name: cainjector + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "cainjector" + app.kubernetes.io/version: "{{ cert_manager_version }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cert-manager-cainjector +subjects: + - name: cert-manager-cainjector + namespace: {{ cert_manager_namespace }} + kind: ServiceAccount +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cert-manager-controller-issuers + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cert-manager-controller-issuers +subjects: + - name: cert-manager + namespace: {{ cert_manager_namespace }} + kind: ServiceAccount +--- +# Source: cert-manager/deploy/charts/cert-manager/templates//rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cert-manager-controller-clusterissuers + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cert-manager-controller-clusterissuers +subjects: + - name: cert-manager + namespace: {{ cert_manager_namespace }} + kind: ServiceAccount +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cert-manager-controller-certificates + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cert-manager-controller-certificates +subjects: + - name: cert-manager + namespace: {{ cert_manager_namespace }} + kind: ServiceAccount +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cert-manager-controller-orders + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cert-manager-controller-orders +subjects: + - name: cert-manager + namespace: {{ cert_manager_namespace }} + kind: ServiceAccount +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cert-manager-controller-challenges + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cert-manager-controller-challenges +subjects: + - name: cert-manager + namespace: {{ cert_manager_namespace }} + kind: ServiceAccount +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cert-manager-controller-ingress-shim + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cert-manager-controller-ingress-shim +subjects: + - name: cert-manager + namespace: {{ cert_manager_namespace }} + kind: ServiceAccount +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cert-manager-controller-approve:cert-manager-io + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "cert-manager" + app.kubernetes.io/version: "{{ cert_manager_version }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cert-manager-controller-approve:cert-manager-io +subjects: + - name: cert-manager + namespace: {{ cert_manager_namespace }} + kind: ServiceAccount +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cert-manager-controller-certificatesigningrequests + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "cert-manager" + app.kubernetes.io/version: "{{ cert_manager_version }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cert-manager-controller-certificatesigningrequests +subjects: + - name: cert-manager + namespace: {{ cert_manager_namespace }} + kind: ServiceAccount +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/webhook-rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cert-manager-webhook:subjectaccessreviews + labels: + app: webhook + app.kubernetes.io/name: webhook + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "webhook" + app.kubernetes.io/version: "{{ cert_manager_version }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cert-manager-webhook:subjectaccessreviews +subjects: +- apiGroup: "" + kind: ServiceAccount + name: cert-manager-webhook + namespace: {{ cert_manager_namespace }} +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/cainjector-rbac.yaml +# leader election rules +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: cert-manager-cainjector:leaderelection + namespace: {{ cert_manager_leader_election_namespace }} + labels: + app: cainjector + app.kubernetes.io/name: cainjector + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "cainjector" + app.kubernetes.io/version: "{{ cert_manager_version }}" +rules: + # Used for leader election by the controller + # cert-manager-cainjector-leader-election is used by the CertificateBased injector controller + # see cmd/cainjector/start.go#L113 + # cert-manager-cainjector-leader-election-core is used by the SecretBased injector controller + # see cmd/cainjector/start.go#L137 + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + resourceNames: ["cert-manager-cainjector-leader-election", "cert-manager-cainjector-leader-election-core"] + verbs: ["get", "update", "patch"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["create"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: cert-manager:leaderelection + namespace: {{ cert_manager_leader_election_namespace }} + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +rules: + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + resourceNames: ["cert-manager-controller"] + verbs: ["get", "update", "patch"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["create"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/webhook-rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: cert-manager-webhook:dynamic-serving + namespace: {{ cert_manager_namespace }} + labels: + app: webhook + app.kubernetes.io/name: webhook + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "webhook" + app.kubernetes.io/version: "{{ cert_manager_version }}" +rules: +- apiGroups: [""] + resources: ["secrets"] + resourceNames: + - 'cert-manager-webhook-ca' + verbs: ["get", "list", "watch", "update"] +# It's not possible to grant CREATE permission on a single resourceName. +- apiGroups: [""] + resources: ["secrets"] + verbs: ["create"] +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/cainjector-rbac.yaml +# grant cert-manager permission to manage the leaderelection configmap in the +# leader election namespace +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: cert-manager-cainjector:leaderelection + namespace: {{ cert_manager_leader_election_namespace }} + labels: + app: cainjector + app.kubernetes.io/name: cainjector + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "cainjector" + app.kubernetes.io/version: "{{ cert_manager_version }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: cert-manager-cainjector:leaderelection +subjects: + - kind: ServiceAccount + name: cert-manager-cainjector + namespace: {{ cert_manager_namespace }} +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/rbac.yaml +# grant cert-manager permission to manage the leaderelection configmap in the +# leader election namespace +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: cert-manager:leaderelection + namespace: {{ cert_manager_leader_election_namespace }} + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: cert-manager:leaderelection +subjects: + - apiGroup: "" + kind: ServiceAccount + name: cert-manager + namespace: {{ cert_manager_namespace }} +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/webhook-rbac.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: cert-manager-webhook:dynamic-serving + namespace: {{ cert_manager_namespace }} + labels: + app: webhook + app.kubernetes.io/name: webhook + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "webhook" + app.kubernetes.io/version: "{{ cert_manager_version }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: cert-manager-webhook:dynamic-serving +subjects: +- apiGroup: "" + kind: ServiceAccount + name: cert-manager-webhook + namespace: {{ cert_manager_namespace }} +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/service.yaml +apiVersion: v1 +kind: Service +metadata: + name: cert-manager + namespace: {{ cert_manager_namespace }} + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +spec: + type: ClusterIP + ports: + - protocol: TCP + port: 9402 + name: tcp-prometheus-servicemonitor + targetPort: 9402 + selector: + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/webhook-service.yaml +apiVersion: v1 +kind: Service +metadata: + name: cert-manager-webhook + namespace: {{ cert_manager_namespace }} + labels: + app: webhook + app.kubernetes.io/name: webhook + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "webhook" + app.kubernetes.io/version: "{{ cert_manager_version }}" +spec: + type: ClusterIP + ports: + - name: https + port: 443 + protocol: TCP + targetPort: "https" + selector: + app.kubernetes.io/name: webhook + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "webhook" +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/cainjector-deployment.yaml +apiVersion: apps/v1 +kind: Deployment +metadata: + name: cert-manager-cainjector + namespace: {{ cert_manager_namespace }} + labels: + app: cainjector + app.kubernetes.io/name: cainjector + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "cainjector" + app.kubernetes.io/version: "{{ cert_manager_version }}" +spec: + replicas: 1 + selector: + matchLabels: + app.kubernetes.io/name: cainjector + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "cainjector" + template: + metadata: + labels: + app: cainjector + app.kubernetes.io/name: cainjector + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "cainjector" + app.kubernetes.io/version: "{{ cert_manager_version }}" + spec: + serviceAccountName: cert-manager-cainjector + securityContext: + runAsNonRoot: true + seccompProfile: + type: RuntimeDefault + containers: + - name: cert-manager-cainjector + image: "{{ cert_manager_cainjector_image_repo }}:{{ cert_manager_cainjector_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - --v=2 + - --leader-election-namespace={{ cert_manager_leader_election_namespace }} + env: + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace +{% if cert_manager_http_proxy is defined and cert_manager_http_proxy != "" %} + - name: HTTP_PROXY + value: "{{ cert_manager_http_proxy }}" +{% endif %} +{% if cert_manager_https_proxy is defined and cert_manager_https_proxy != "" %} + - name: HTTPS_PROXY + value: "{{ cert_manager_https_proxy }}" +{% endif %} +{% if cert_manager_no_proxy is defined and cert_manager_no_proxy != "" %} + - name: NO_PROXY + value: "{{ cert_manager_no_proxy }}" +{% endif %} + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - ALL + runAsNonRoot: true + seccompProfile: + type: RuntimeDefault +{% if cert_manager_tolerations %} + tolerations: + {{ cert_manager_tolerations | to_nice_yaml(indent=2) | indent(width=8) }} +{% endif %} +{% if cert_manager_nodeselector %} + nodeSelector: + {{ cert_manager_nodeselector | to_nice_yaml | indent(width=8) }} +{% endif %} +{% if cert_manager_affinity %} + affinity: + {{ cert_manager_affinity | to_nice_yaml | indent(width=8) }} +{% endif %} +--- +{% if cert_manager_trusted_internal_ca is defined %} +apiVersion: v1 +data: + internal-ca.pem: | + {{ cert_manager_trusted_internal_ca | indent(width=4, first=False) }} +kind: ConfigMap +metadata: + name: ca-internal-truststore + namespace: {{ cert_manager_namespace }} +--- +{% endif %} +# Source: cert-manager/deploy/charts/cert-manager/templates/deployment.yaml +apiVersion: apps/v1 +kind: Deployment +metadata: + name: cert-manager + namespace: {{ cert_manager_namespace }} + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" +spec: + replicas: 1 + selector: + matchLabels: + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + template: + metadata: + labels: + app: cert-manager + app.kubernetes.io/name: cert-manager + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "controller" + app.kubernetes.io/version: "{{ cert_manager_version }}" + annotations: + prometheus.io/path: "/metrics" + prometheus.io/scrape: 'true' + prometheus.io/port: '9402' + spec: + serviceAccountName: cert-manager + securityContext: + runAsNonRoot: true + seccompProfile: + type: RuntimeDefault + containers: + - name: cert-manager-controller + image: "{{ cert_manager_controller_image_repo }}:{{ cert_manager_controller_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - --v=2 + - --cluster-resource-namespace=$(POD_NAMESPACE) + - --leader-election-namespace={{ cert_manager_leader_election_namespace }} +{% for extra_arg in cert_manager_controller_extra_args %} + - {{ extra_arg }} +{% endfor %} + ports: + - containerPort: 9402 + name: http-metrics + protocol: TCP + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - ALL + runAsNonRoot: true + seccompProfile: + type: RuntimeDefault + env: + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace +{% if cert_manager_http_proxy is defined and cert_manager_http_proxy != "" %} + - name: HTTP_PROXY + value: "{{ cert_manager_http_proxy }}" +{% endif %} +{% if cert_manager_https_proxy is defined and cert_manager_https_proxy != "" %} + - name: HTTPS_PROXY + value: "{{ cert_manager_https_proxy }}" +{% endif %} +{% if cert_manager_no_proxy is defined and cert_manager_no_proxy != "" %} + - name: NO_PROXY + value: "{{ cert_manager_no_proxy }}" +{% endif %} +{% if cert_manager_trusted_internal_ca is defined %} + volumeMounts: + - mountPath: /etc/ssl/certs/internal-ca.pem + name: ca-internal-truststore + subPath: internal-ca.pem + volumes: + - configMap: + defaultMode: 420 + name: ca-internal-truststore + name: ca-internal-truststore +{% endif %} +{% if cert_manager_tolerations %} + tolerations: + {{ cert_manager_tolerations | to_nice_yaml(indent=2) | indent(width=8) }} +{% endif %} +{% if cert_manager_nodeselector %} + nodeSelector: + {{ cert_manager_nodeselector | to_nice_yaml | indent(width=8) }} +{% endif %} +{% if cert_manager_affinity %} + affinity: + {{ cert_manager_affinity | to_nice_yaml | indent(width=8) }} +{% endif %} +{% if cert_manager_dns_policy %} + dnsPolicy: {{ cert_manager_dns_policy }} +{% endif %} +{% if cert_manager_dns_config %} + dnsConfig: + {{ cert_manager_dns_config | to_nice_yaml | indent(width=8) }} +{% endif %} +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/webhook-deployment.yaml +apiVersion: apps/v1 +kind: Deployment +metadata: + name: cert-manager-webhook + namespace: {{ cert_manager_namespace }} + labels: + app: webhook + app.kubernetes.io/name: webhook + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "webhook" + app.kubernetes.io/version: "{{ cert_manager_version }}" +spec: + replicas: 1 + selector: + matchLabels: + app.kubernetes.io/name: webhook + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "webhook" + template: + metadata: + labels: + app: webhook + app.kubernetes.io/name: webhook + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "webhook" + app.kubernetes.io/version: "{{ cert_manager_version }}" + spec: + serviceAccountName: cert-manager-webhook + securityContext: + runAsNonRoot: true + seccompProfile: + type: RuntimeDefault + containers: + - name: cert-manager-webhook + image: "{{ cert_manager_webhook_image_repo }}:{{ cert_manager_webhook_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - --v=2 + - --secure-port=10250 + - --dynamic-serving-ca-secret-namespace=$(POD_NAMESPACE) + - --dynamic-serving-ca-secret-name=cert-manager-webhook-ca + - --dynamic-serving-dns-names=cert-manager-webhook + - --dynamic-serving-dns-names=cert-manager-webhook.$(POD_NAMESPACE) + - --dynamic-serving-dns-names=cert-manager-webhook.$(POD_NAMESPACE).svc + ports: + - name: https + protocol: TCP + containerPort: 10250 + - name: healthcheck + protocol: TCP + containerPort: 6080 + livenessProbe: + httpGet: + path: /livez + port: 6080 + scheme: HTTP + initialDelaySeconds: 60 + periodSeconds: 10 + timeoutSeconds: 1 + successThreshold: 1 + failureThreshold: 3 + readinessProbe: + httpGet: + path: /healthz + port: 6080 + scheme: HTTP + initialDelaySeconds: 5 + periodSeconds: 5 + timeoutSeconds: 1 + successThreshold: 1 + failureThreshold: 3 + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - ALL + runAsNonRoot: true + seccompProfile: + type: RuntimeDefault + env: + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace +{% if cert_manager_http_proxy is defined and cert_manager_http_proxy != "" %} + - name: HTTP_PROXY + value: "{{ cert_manager_http_proxy }}" +{% endif %} +{% if cert_manager_https_proxy is defined and cert_manager_https_proxy != "" %} + - name: HTTPS_PROXY + value: "{{ cert_manager_https_proxy }}" +{% endif %} +{% if cert_manager_no_proxy is defined and cert_manager_no_proxy != "" %} + - name: NO_PROXY + value: "{{ cert_manager_no_proxy }}" +{% endif %} +{% if cert_manager_tolerations %} + tolerations: + {{ cert_manager_tolerations | to_nice_yaml(indent=2) | indent(width=8) }} +{% endif %} +{% if cert_manager_nodeselector %} + nodeSelector: + {{ cert_manager_nodeselector | to_nice_yaml | indent(width=8) }} +{% endif %} +{% if cert_manager_affinity %} + affinity: + {{ cert_manager_affinity | to_nice_yaml | indent(width=8) }} +{% endif %} +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/webhook-mutating-webhook.yaml +apiVersion: admissionregistration.k8s.io/v1 +kind: MutatingWebhookConfiguration +metadata: + name: cert-manager-webhook + labels: + app: webhook + app.kubernetes.io/name: webhook + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "webhook" + app.kubernetes.io/version: "{{ cert_manager_version }}" + annotations: + cert-manager.io/inject-ca-from-secret: "{{ cert_manager_namespace }}/cert-manager-webhook-ca" +webhooks: + - name: webhook.cert-manager.io + rules: + - apiGroups: + - "cert-manager.io" + - "acme.cert-manager.io" + apiVersions: + - "v1" + operations: + - CREATE + - UPDATE + resources: + - "*/*" + admissionReviewVersions: ["v1"] + # This webhook only accepts v1 cert-manager resources. + # Equivalent matchPolicy ensures that non-v1 resource requests are sent to + # this webhook (after the resources have been converted to v1). + matchPolicy: Equivalent + timeoutSeconds: 10 + failurePolicy: Fail + # Only include 'sideEffects' field in Kubernetes 1.12+ + sideEffects: None + clientConfig: + service: + name: cert-manager-webhook + namespace: {{ cert_manager_namespace }} + path: /mutate +--- +# Source: cert-manager/deploy/charts/cert-manager/templates/webhook-validating-webhook.yaml +apiVersion: admissionregistration.k8s.io/v1 +kind: ValidatingWebhookConfiguration +metadata: + name: cert-manager-webhook + labels: + app: webhook + app.kubernetes.io/name: webhook + app.kubernetes.io/instance: cert-manager + app.kubernetes.io/component: "webhook" + app.kubernetes.io/version: "{{ cert_manager_version }}" + annotations: + cert-manager.io/inject-ca-from-secret: "{{ cert_manager_namespace }}/cert-manager-webhook-ca" +webhooks: + - name: webhook.cert-manager.io + namespaceSelector: + matchExpressions: + - key: "cert-manager.io/disable-validation" + operator: "NotIn" + values: + - "true" + - key: "name" + operator: "NotIn" + values: + - cert-manager + rules: + - apiGroups: + - "cert-manager.io" + - "acme.cert-manager.io" + apiVersions: + - "v1" + operations: + - CREATE + - UPDATE + resources: + - "*/*" + admissionReviewVersions: ["v1"] + # This webhook only accepts v1 cert-manager resources. + # Equivalent matchPolicy ensures that non-v1 resource requests are sent to + # this webhook (after the resources have been converted to v1). + matchPolicy: Equivalent + timeoutSeconds: 10 + failurePolicy: Fail + sideEffects: None + clientConfig: + service: + name: cert-manager-webhook + namespace: {{ cert_manager_namespace }} + path: /validate diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/defaults/main.yml b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/defaults/main.yml new file mode 100644 index 0000000..7a5c134 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/defaults/main.yml @@ -0,0 +1,22 @@ +--- +ingress_nginx_namespace: "ingress-nginx" +ingress_nginx_host_network: false +ingress_publish_status_address: "" +ingress_nginx_nodeselector: + kubernetes.io/os: "linux" +ingress_nginx_tolerations: [] +ingress_nginx_insecure_port: 80 +ingress_nginx_secure_port: 443 +ingress_nginx_metrics_port: 10254 +ingress_nginx_configmap: {} +ingress_nginx_configmap_tcp_services: {} +ingress_nginx_configmap_udp_services: {} +ingress_nginx_extra_args: [] +ingress_nginx_termination_grace_period_seconds: 300 +ingress_nginx_class: nginx +ingress_nginx_without_class: true +ingress_nginx_default: false +ingress_nginx_webhook_enabled: false +ingress_nginx_webhook_job_ttl: 1800 + +ingress_nginx_probe_initial_delay_seconds: 10 diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/tasks/main.yml b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/tasks/main.yml new file mode 100644 index 0000000..b67a17f --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/tasks/main.yml @@ -0,0 +1,61 @@ +--- + +- name: NGINX Ingress Controller | Create addon dir + file: + path: "{{ kube_config_dir }}/addons/ingress_nginx" + state: directory + owner: root + group: root + mode: 0755 + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: NGINX Ingress Controller | Templates list + set_fact: + ingress_nginx_templates: + - { name: 00-namespace, file: 00-namespace.yml, type: ns } + - { name: cm-ingress-nginx, file: cm-ingress-nginx.yml, type: cm } + - { name: cm-tcp-services, file: cm-tcp-services.yml, type: cm } + - { name: cm-udp-services, file: cm-udp-services.yml, type: cm } + - { name: sa-ingress-nginx, file: sa-ingress-nginx.yml, type: sa } + - { name: clusterrole-ingress-nginx, file: clusterrole-ingress-nginx.yml, type: clusterrole } + - { name: clusterrolebinding-ingress-nginx, file: clusterrolebinding-ingress-nginx.yml, type: clusterrolebinding } + - { name: role-ingress-nginx, file: role-ingress-nginx.yml, type: role } + - { name: rolebinding-ingress-nginx, file: rolebinding-ingress-nginx.yml, type: rolebinding } + - { name: ingressclass-nginx, file: ingressclass-nginx.yml, type: ingressclass } + - { name: ds-ingress-nginx-controller, file: ds-ingress-nginx-controller.yml, type: ds } + ingress_nginx_templates_for_webhook: + - { name: admission-webhook-configuration, file: admission-webhook-configuration.yml, type: sa } + - { name: sa-admission-webhook, file: sa-admission-webhook.yml, type: sa } + - { name: clusterrole-admission-webhook, file: clusterrole-admission-webhook.yml, type: clusterrole } + - { name: clusterrolebinding-admission-webhook, file: clusterrolebinding-admission-webhook.yml, type: clusterrolebinding } + - { name: role-admission-webhook, file: role-admission-webhook.yml, type: role } + - { name: rolebinding-admission-webhook, file: rolebinding-admission-webhook.yml, type: rolebinding } + - { name: admission-webhook-job, file: admission-webhook-job.yml, type: job } + +- name: NGINX Ingress Controller | Append extra templates to NGINX Ingress Templates list for webhook + set_fact: + ingress_nginx_templates: "{{ ingress_nginx_templates + ingress_nginx_templates_for_webhook }}" + when: ingress_nginx_webhook_enabled + +- name: NGINX Ingress Controller | Create manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/ingress_nginx/{{ item.file }}" + mode: 0644 + with_items: "{{ ingress_nginx_templates }}" + register: ingress_nginx_manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: NGINX Ingress Controller | Apply manifests + kube: + name: "{{ item.item.name }}" + namespace: "{{ ingress_nginx_namespace }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/ingress_nginx/{{ item.item.file }}" + state: "latest" + with_items: "{{ ingress_nginx_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/00-namespace.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/00-namespace.yml.j2 new file mode 100644 index 0000000..1f12366 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/00-namespace.yml.j2 @@ -0,0 +1,7 @@ +--- +apiVersion: v1 +kind: Namespace +metadata: + name: {{ ingress_nginx_namespace }} + labels: + name: {{ ingress_nginx_namespace }} diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/admission-webhook-configuration.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/admission-webhook-configuration.yml.j2 new file mode 100644 index 0000000..d6878a0 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/admission-webhook-configuration.yml.j2 @@ -0,0 +1,29 @@ +apiVersion: admissionregistration.k8s.io/v1 +kind: ValidatingWebhookConfiguration +metadata: + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx + name: ingress-nginx-admission +webhooks: +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: ingress-nginx-controller-admission + namespace: {{ ingress_nginx_namespace }} + path: /networking/v1/ingresses + failurePolicy: Fail + matchPolicy: Equivalent + name: validate.nginx.ingress.kubernetes.io + rules: + - apiGroups: + - networking.k8s.io + apiVersions: + - v1 + operations: + - CREATE + - UPDATE + resources: + - ingresses + sideEffects: None diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/admission-webhook-job.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/admission-webhook-job.yml.j2 new file mode 100644 index 0000000..258a7a1 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/admission-webhook-job.yml.j2 @@ -0,0 +1,86 @@ +--- +apiVersion: batch/v1 +kind: Job +metadata: + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx + name: ingress-nginx-admission-create + namespace: {{ ingress_nginx_namespace }} +spec: + template: + metadata: + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx + name: ingress-nginx-admission-create + spec: + containers: + - args: + - create + - --host=ingress-nginx-controller-admission,ingress-nginx-controller-admission.$(POD_NAMESPACE).svc + - --namespace=$(POD_NAMESPACE) + - --secret-name=ingress-nginx-admission + env: + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + image: "{{ ingress_nginx_kube_webhook_certgen_image_repo }}:{{ ingress_nginx_kube_webhook_certgen_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + name: create + securityContext: + allowPrivilegeEscalation: false + nodeSelector: + kubernetes.io/os: linux + restartPolicy: OnFailure + securityContext: + fsGroup: 2000 + runAsNonRoot: true + runAsUser: 2000 + serviceAccountName: ingress-nginx-admission + ttlSecondsAfterFinished: {{ ingress_nginx_webhook_job_ttl }} +--- +apiVersion: batch/v1 +kind: Job +metadata: + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx + name: ingress-nginx-admission-patch + namespace: {{ ingress_nginx_namespace }} +spec: + template: + metadata: + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx + name: ingress-nginx-admission-patch + spec: + containers: + - args: + - patch + - --webhook-name=ingress-nginx-admission + - --namespace=$(POD_NAMESPACE) + - --patch-mutating=false + - --secret-name=ingress-nginx-admission + - --patch-failure-policy=Fail + env: + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + image: "{{ ingress_nginx_kube_webhook_certgen_image_repo }}:{{ ingress_nginx_kube_webhook_certgen_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + name: patch + securityContext: + allowPrivilegeEscalation: false + nodeSelector: + kubernetes.io/os: linux + restartPolicy: OnFailure + securityContext: + fsGroup: 2000 + runAsNonRoot: true + runAsUser: 2000 + serviceAccountName: ingress-nginx-admission + ttlSecondsAfterFinished: {{ ingress_nginx_webhook_job_ttl }} diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrole-admission-webhook.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrole-admission-webhook.yml.j2 new file mode 100644 index 0000000..daa4753 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrole-admission-webhook.yml.j2 @@ -0,0 +1,15 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx + name: ingress-nginx-admission +rules: +- apiGroups: + - admissionregistration.k8s.io + resources: + - validatingwebhookconfigurations + verbs: + - get + - update diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrole-ingress-nginx.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrole-ingress-nginx.yml.j2 new file mode 100644 index 0000000..38118bf --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrole-ingress-nginx.yml.j2 @@ -0,0 +1,36 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: ingress-nginx + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx +rules: + - apiGroups: [""] + resources: ["configmaps", "endpoints", "nodes", "pods", "secrets", "namespaces"] + verbs: ["list", "watch"] + - apiGroups: [""] + resources: ["nodes"] + verbs: ["get"] + - apiGroups: [""] + resources: ["services"] + verbs: ["get", "list", "watch"] + - apiGroups: ["networking.k8s.io"] + resources: ["ingresses"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["create", "patch"] + - apiGroups: ["extensions","networking.k8s.io"] + resources: ["ingresses/status"] + verbs: ["update"] + - apiGroups: ["networking.k8s.io"] + resources: ["ingressclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["list", "watch"] + - apiGroups: ["discovery.k8s.io"] + resources: ["endpointslices"] + verbs: ["get", "list", "watch"] diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrolebinding-admission-webhook.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrolebinding-admission-webhook.yml.j2 new file mode 100644 index 0000000..8791594 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrolebinding-admission-webhook.yml.j2 @@ -0,0 +1,16 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx + name: ingress-nginx-admission +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: ingress-nginx-admission +subjects: + - kind: ServiceAccount + name: ingress-nginx-admission + namespace: {{ ingress_nginx_namespace }} diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrolebinding-ingress-nginx.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrolebinding-ingress-nginx.yml.j2 new file mode 100644 index 0000000..ad83dc2 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/clusterrolebinding-ingress-nginx.yml.j2 @@ -0,0 +1,16 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: ingress-nginx + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: ingress-nginx +subjects: + - kind: ServiceAccount + name: ingress-nginx + namespace: {{ ingress_nginx_namespace }} diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/cm-ingress-nginx.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/cm-ingress-nginx.yml.j2 new file mode 100644 index 0000000..9f1e3bb --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/cm-ingress-nginx.yml.j2 @@ -0,0 +1,13 @@ +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: ingress-nginx + namespace: {{ ingress_nginx_namespace }} + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx +{% if ingress_nginx_configmap %} +data: + {{ ingress_nginx_configmap | to_nice_yaml | indent(2) }} +{%- endif %} diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/cm-tcp-services.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/cm-tcp-services.yml.j2 new file mode 100644 index 0000000..9752081 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/cm-tcp-services.yml.j2 @@ -0,0 +1,13 @@ +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: tcp-services + namespace: {{ ingress_nginx_namespace }} + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx +{% if ingress_nginx_configmap_tcp_services %} +data: + {{ ingress_nginx_configmap_tcp_services | to_nice_yaml | indent(2) }} +{%- endif %} diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/cm-udp-services.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/cm-udp-services.yml.j2 new file mode 100644 index 0000000..a3f6613 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/cm-udp-services.yml.j2 @@ -0,0 +1,13 @@ +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: udp-services + namespace: {{ ingress_nginx_namespace }} + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx +{% if ingress_nginx_configmap_udp_services %} +data: + {{ ingress_nginx_configmap_udp_services | to_nice_yaml | indent(2) }} +{%- endif %} diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/ds-ingress-nginx-controller.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/ds-ingress-nginx-controller.yml.j2 new file mode 100644 index 0000000..70e4ea0 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/ds-ingress-nginx-controller.yml.j2 @@ -0,0 +1,140 @@ +--- +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: ingress-nginx-controller + namespace: {{ ingress_nginx_namespace }} + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx +spec: + selector: + matchLabels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx + template: + metadata: + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx + annotations: + prometheus.io/port: "10254" + prometheus.io/scrape: "true" + spec: + serviceAccountName: ingress-nginx + terminationGracePeriodSeconds: {{ ingress_nginx_termination_grace_period_seconds }} +{% if ingress_nginx_host_network %} + hostNetwork: true + dnsPolicy: ClusterFirstWithHostNet +{% endif %} +{% if ingress_nginx_nodeselector %} + nodeSelector: + {{ ingress_nginx_nodeselector | to_nice_yaml | indent(width=8) }} +{%- endif %} +{% if ingress_nginx_tolerations %} + tolerations: + {{ ingress_nginx_tolerations | to_nice_yaml(indent=2) | indent(width=8) }} +{% endif %} + priorityClassName: {% if ingress_nginx_namespace == 'kube-system' %}system-node-critical{% else %}k8s-cluster-critical{% endif %}{{ '' }} + containers: + - name: ingress-nginx-controller + image: {{ ingress_nginx_controller_image_repo }}:{{ ingress_nginx_controller_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + lifecycle: + preStop: + exec: + command: + - /wait-shutdown + args: + - /nginx-ingress-controller + - --configmap=$(POD_NAMESPACE)/ingress-nginx + - --tcp-services-configmap=$(POD_NAMESPACE)/tcp-services + - --udp-services-configmap=$(POD_NAMESPACE)/udp-services + - --annotations-prefix=nginx.ingress.kubernetes.io + - --ingress-class={{ ingress_nginx_class }} +{% if ingress_nginx_without_class %} + - --watch-ingress-without-class=true +{% endif %} +{% if ingress_nginx_host_network %} + - --report-node-internal-ip-address +{% endif %} +{% if ingress_publish_status_address != "" %} + - --publish-status-address={{ ingress_publish_status_address }} +{% endif %} +{% for extra_arg in ingress_nginx_extra_args %} + - {{ extra_arg }} +{% endfor %} +{% if ingress_nginx_webhook_enabled %} + - --validating-webhook=:8443 + - --validating-webhook-certificate=/usr/local/certificates/cert + - --validating-webhook-key=/usr/local/certificates/key +{% endif %} + securityContext: + capabilities: + drop: + - ALL + add: + - NET_BIND_SERVICE + # www-data -> 101 + runAsUser: 101 + allowPrivilegeEscalation: true + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + - name: LD_PRELOAD + value: /usr/local/lib/libmimalloc.so + ports: + - name: http + containerPort: 80 + hostPort: {{ ingress_nginx_insecure_port }} + - name: https + containerPort: 443 + hostPort: {{ ingress_nginx_secure_port }} + - name: metrics + containerPort: 10254 +{% if not ingress_nginx_host_network %} + hostPort: {{ ingress_nginx_metrics_port }} +{% endif %} +{% if ingress_nginx_webhook_enabled %} + - name: webhook + containerPort: 8443 + protocol: TCP +{% endif %} + livenessProbe: + httpGet: + path: /healthz + port: 10254 + scheme: HTTP + initialDelaySeconds: {{ ingress_nginx_probe_initial_delay_seconds }} + periodSeconds: 10 + timeoutSeconds: 5 + successThreshold: 1 + failureThreshold: 3 + readinessProbe: + httpGet: + path: /healthz + port: 10254 + scheme: HTTP + initialDelaySeconds: {{ ingress_nginx_probe_initial_delay_seconds }} + periodSeconds: 10 + timeoutSeconds: 5 + successThreshold: 1 + failureThreshold: 3 +{% if ingress_nginx_webhook_enabled %} + volumeMounts: + - mountPath: /usr/local/certificates/ + name: webhook-cert + readOnly: true +{% endif %} +{% if ingress_nginx_webhook_enabled %} + volumes: + - name: webhook-cert + secret: + secretName: ingress-nginx-admission +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/ingressclass-nginx.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/ingressclass-nginx.yml.j2 new file mode 100644 index 0000000..0e5fa8c --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/ingressclass-nginx.yml.j2 @@ -0,0 +1,13 @@ +apiVersion: networking.k8s.io/v1 +kind: IngressClass +metadata: + name: {{ ingress_nginx_class }} + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx +{% if ingress_nginx_default %} + annotations: + ingressclass.kubernetes.io/is-default-class: "true" +{% endif %} +spec: + controller: k8s.io/ingress-nginx diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/role-admission-webhook.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/role-admission-webhook.yml.j2 new file mode 100644 index 0000000..5d1bb01 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/role-admission-webhook.yml.j2 @@ -0,0 +1,17 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx + name: ingress-nginx-admission + namespace: {{ ingress_nginx_namespace }} +rules: +- apiGroups: + - "" + resources: + - secrets + verbs: + - get + - create diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/role-ingress-nginx.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/role-ingress-nginx.yml.j2 new file mode 100644 index 0000000..6c4b1c1 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/role-ingress-nginx.yml.j2 @@ -0,0 +1,53 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: ingress-nginx + namespace: {{ ingress_nginx_namespace }} + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx +rules: + - apiGroups: [""] + resources: ["namespaces"] + verbs: ["get"] + - apiGroups: [""] + resources: ["configmaps", "pods", "secrets", "endpoints"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["services"] + verbs: ["get", "list", "watch"] + - apiGroups: ["networking.k8s.io"] + resources: ["ingresses"] + verbs: ["get", "list", "watch"] + - apiGroups: ["networking.k8s.io"] + resources: ["ingresses/status"] + verbs: ["update"] + - apiGroups: ["networking.k8s.io"] + resources: ["ingressclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + # Defaults to "-" + # Here: "-" + # This has to be adapted if you change either parameter + # when launching the nginx-ingress-controller. + resourceNames: ["ingress-controller-leader-{{ ingress_nginx_class }}"] + verbs: ["get", "update"] + - apiGroups: [""] + resources: ["events"] + verbs: ["create", "patch"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + # Defaults to "-" + # Here: "-" + # This has to be adapted if you change either parameter + # when launching the nginx-ingress-controller. + resourceNames: ["ingress-controller-leader-{{ ingress_nginx_class }}"] + verbs: ["get", "update"] + - apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["create"] + - apiGroups: ["discovery.k8s.io"] + resources: ["endpointslices"] + verbs: ["get", "list", "watch"] diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/rolebinding-admission-webhook.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/rolebinding-admission-webhook.yml.j2 new file mode 100644 index 0000000..671912d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/rolebinding-admission-webhook.yml.j2 @@ -0,0 +1,17 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx + name: ingress-nginx-admission + namespace: {{ ingress_nginx_namespace }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: ingress-nginx-admission +subjects: +- kind: ServiceAccount + name: ingress-nginx-admission + namespace: {{ ingress_nginx_namespace }} diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/rolebinding-ingress-nginx.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/rolebinding-ingress-nginx.yml.j2 new file mode 100644 index 0000000..142d400 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/rolebinding-ingress-nginx.yml.j2 @@ -0,0 +1,17 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: ingress-nginx + namespace: {{ ingress_nginx_namespace }} + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: ingress-nginx +subjects: + - kind: ServiceAccount + name: ingress-nginx + namespace: {{ ingress_nginx_namespace }} diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/sa-admission-webhook.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/sa-admission-webhook.yml.j2 new file mode 100644 index 0000000..488a045 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/sa-admission-webhook.yml.j2 @@ -0,0 +1,8 @@ +apiVersion: v1 +kind: ServiceAccount +metadata: + name: ingress-nginx-admission + namespace: {{ ingress_nginx_namespace }} + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/sa-ingress-nginx.yml.j2 b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/sa-ingress-nginx.yml.j2 new file mode 100644 index 0000000..305d553 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/ingress_nginx/templates/sa-ingress-nginx.yml.j2 @@ -0,0 +1,9 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: ingress-nginx + namespace: {{ ingress_nginx_namespace }} + labels: + app.kubernetes.io/name: ingress-nginx + app.kubernetes.io/part-of: ingress-nginx diff --git a/kubespray/roles/kubernetes-apps/ingress_controller/meta/main.yml b/kubespray/roles/kubernetes-apps/ingress_controller/meta/main.yml new file mode 100644 index 0000000..b269607 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/ingress_controller/meta/main.yml @@ -0,0 +1,22 @@ +--- +dependencies: + - role: kubernetes-apps/ingress_controller/ingress_nginx + when: ingress_nginx_enabled + tags: + - apps + - ingress-controller + - ingress-nginx + + - role: kubernetes-apps/ingress_controller/cert_manager + when: cert_manager_enabled + tags: + - apps + - ingress-controller + - cert-manager + + - role: kubernetes-apps/ingress_controller/alb_ingress_controller + when: ingress_alb_enabled + tags: + - apps + - ingress-controller + - ingress_alb diff --git a/kubespray/roles/kubernetes-apps/krew/defaults/main.yml b/kubespray/roles/kubernetes-apps/krew/defaults/main.yml new file mode 100644 index 0000000..6878427 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/krew/defaults/main.yml @@ -0,0 +1,5 @@ +--- +krew_enabled: false +krew_root_dir: "/usr/local/krew" +krew_default_index_uri: https://github.com/kubernetes-sigs/krew-index.git +krew_no_upgrade_check: 0 diff --git a/kubespray/roles/kubernetes-apps/krew/tasks/krew.yml b/kubespray/roles/kubernetes-apps/krew/tasks/krew.yml new file mode 100644 index 0000000..a8b5201 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/krew/tasks/krew.yml @@ -0,0 +1,38 @@ +--- +- name: Krew | Download krew + include_tasks: "../../../download/tasks/download_file.yml" + vars: + download: "{{ download_defaults | combine(downloads.krew) }}" + +- name: Krew | krew env + template: + src: krew.j2 + dest: /etc/bash_completion.d/krew + mode: 0644 + +- name: Krew | Copy krew manifest + template: + src: krew.yml.j2 + dest: "{{ local_release_dir }}/krew.yml" + mode: 0644 + +- name: Krew | Install krew # noqa command-instead-of-shell + shell: "{{ local_release_dir }}/krew-{{ host_os }}_{{ image_arch }} install --archive={{ local_release_dir }}/krew-{{ host_os }}_{{ image_arch }}.tar.gz --manifest={{ local_release_dir }}/krew.yml" + environment: + KREW_ROOT: "{{ krew_root_dir }}" + KREW_DEFAULT_INDEX_URI: "{{ krew_default_index_uri | default('') }}" + +- name: Krew | Get krew completion + command: "{{ local_release_dir }}/krew-{{ host_os }}_{{ image_arch }} completion bash" + changed_when: False + register: krew_completion + check_mode: False + ignore_errors: yes # noqa ignore-errors + +- name: Krew | Install krew completion + copy: + dest: /etc/bash_completion.d/krew.sh + content: "{{ krew_completion.stdout }}" + mode: 0755 + become: True + when: krew_completion.rc == 0 diff --git a/kubespray/roles/kubernetes-apps/krew/tasks/main.yml b/kubespray/roles/kubernetes-apps/krew/tasks/main.yml new file mode 100644 index 0000000..40729e8 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/krew/tasks/main.yml @@ -0,0 +1,10 @@ +--- +- name: Krew | install krew on kube_control_plane + import_tasks: krew.yml + +- name: Krew | install krew on localhost + import_tasks: krew.yml + delegate_to: localhost + connection: local + run_once: true + when: kubectl_localhost diff --git a/kubespray/roles/kubernetes-apps/krew/templates/krew.j2 b/kubespray/roles/kubernetes-apps/krew/templates/krew.j2 new file mode 100644 index 0000000..62a744c --- /dev/null +++ b/kubespray/roles/kubernetes-apps/krew/templates/krew.j2 @@ -0,0 +1,7 @@ +# krew bash env(kubespray) +export KREW_ROOT="{{ krew_root_dir }}" +{% if krew_default_index_uri is defined %} +export KREW_DEFAULT_INDEX_URI='{{ krew_default_index_uri }}' +{% endif %} +export PATH="${KREW_ROOT:-$HOME/.krew}/bin:$PATH" +export KREW_NO_UPGRADE_CHECK={{ krew_no_upgrade_check }} diff --git a/kubespray/roles/kubernetes-apps/krew/templates/krew.yml.j2 b/kubespray/roles/kubernetes-apps/krew/templates/krew.yml.j2 new file mode 100644 index 0000000..b0c6152 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/krew/templates/krew.yml.j2 @@ -0,0 +1,100 @@ +apiVersion: krew.googlecontainertools.github.com/v1alpha2 +kind: Plugin +metadata: + name: krew +spec: + version: "{{ krew_version }}" + homepage: https://krew.sigs.k8s.io/ + shortDescription: Package manager for kubectl plugins. + caveats: | + krew is now installed! To start using kubectl plugins, you need to add + krew's installation directory to your PATH: + + * macOS/Linux: + - Add the following to your ~/.bashrc or ~/.zshrc: + export PATH="${KREW_ROOT:-$HOME/.krew}/bin:$PATH" + - Restart your shell. + + * Windows: Add %USERPROFILE%\.krew\bin to your PATH environment variable + + To list krew commands and to get help, run: + $ kubectl krew + For a full list of available plugins, run: + $ kubectl krew search + + You can find documentation at + https://krew.sigs.k8s.io/docs/user-guide/quickstart/. + + platforms: + - uri: {{ krew_download_url }} + sha256: {{ krew_archive_checksum }} + bin: krew + files: + - from: ./krew-darwin_amd64 + to: krew + - from: ./LICENSE + to: . + selector: + matchLabels: + os: darwin + arch: amd64 + - uri: {{ krew_download_url }} + sha256: {{ krew_archive_checksum }} + bin: krew + files: + - from: ./krew-darwin_arm64 + to: krew + - from: ./LICENSE + to: . + selector: + matchLabels: + os: darwin + arch: arm64 + - uri: {{ krew_download_url }} + sha256: {{ krew_archive_checksum }} + bin: krew + files: + - from: ./krew-linux_amd64 + to: krew + - from: ./LICENSE + to: . + selector: + matchLabels: + os: linux + arch: amd64 + - uri: {{ krew_download_url }} + sha256: {{ krew_archive_checksum }} + bin: krew + files: + - from: ./krew-linux_arm + to: krew + - from: ./LICENSE + to: . + selector: + matchLabels: + os: linux + arch: arm + - uri: {{ krew_download_url }} + sha256: {{ krew_archive_checksum }} + bin: krew + files: + - from: ./krew-linux_arm64 + to: krew + - from: ./LICENSE + to: . + selector: + matchLabels: + os: linux + arch: arm64 + - uri: {{ krew_download_url }} + sha256: {{ krew_archive_checksum }} + bin: krew.exe + files: + - from: ./krew-windows_amd64.exe + to: krew.exe + - from: ./LICENSE + to: . + selector: + matchLabels: + os: windows + arch: amd64 diff --git a/kubespray/roles/kubernetes-apps/kubelet-csr-approver/defaults/main.yml b/kubespray/roles/kubernetes-apps/kubelet-csr-approver/defaults/main.yml new file mode 100644 index 0000000..2edce70 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/kubelet-csr-approver/defaults/main.yml @@ -0,0 +1,12 @@ +--- +kubelet_csr_approver_enabled: "{{ kubelet_rotate_server_certificates }}" +kubelet_csr_approver_namespace: kube-system + +kubelet_csr_approver_repository_name: kubelet-csr-approver +kubelet_csr_approver_repository_url: https://postfinance.github.io/kubelet-csr-approver +kubelet_csr_approver_chart_ref: "{{ kubelet_csr_approver_repository_name }}/kubelet-csr-approver" +kubelet_csr_approver_chart_version: 0.2.8 + +# Fill values override here +# See upstream https://github.com/postfinance/kubelet-csr-approver +kubelet_csr_approver_values: {} diff --git a/kubespray/roles/kubernetes-apps/kubelet-csr-approver/meta/main.yml b/kubespray/roles/kubernetes-apps/kubelet-csr-approver/meta/main.yml new file mode 100644 index 0000000..93d1383 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/kubelet-csr-approver/meta/main.yml @@ -0,0 +1,20 @@ +--- +dependencies: + - role: helm-apps + when: + - inventory_hostname == groups['kube_control_plane'][0] + - kubelet_csr_approver_enabled + environment: + http_proxy: "{{ http_proxy | default('') }}" + https_proxy: "{{ https_proxy | default('') }}" + release_common_opts: {} + releases: + - name: kubelet-csr-approver + namespace: "{{ kubelet_csr_approver_namespace }}" + chart_ref: "{{ kubelet_csr_approver_chart_ref }}" + chart_version: "{{ kubelet_csr_approver_chart_version }}" + wait: true + values: "{{ kubelet_csr_approver_values }}" + repositories: + - name: "{{ kubelet_csr_approver_repository_name }}" + url: "{{ kubelet_csr_approver_repository_url }}" diff --git a/kubespray/roles/kubernetes-apps/meta/main.yml b/kubespray/roles/kubernetes-apps/meta/main.yml new file mode 100644 index 0000000..9c19fde --- /dev/null +++ b/kubespray/roles/kubernetes-apps/meta/main.yml @@ -0,0 +1,126 @@ +--- +dependencies: + - role: kubernetes-apps/ansible + when: + - inventory_hostname == groups['kube_control_plane'][0] + + - role: kubernetes-apps/helm + when: + - helm_enabled + tags: + - helm + + - role: kubernetes-apps/krew + when: + - krew_enabled + tags: + - krew + + - role: kubernetes-apps/registry + when: + - registry_enabled + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - registry + + - role: kubernetes-apps/metrics_server + when: + - metrics_server_enabled + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - metrics_server + + - role: kubernetes-apps/csi_driver/csi_crd + when: + - cinder_csi_enabled or csi_snapshot_controller_enabled + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - csi-driver + + - role: kubernetes-apps/csi_driver/cinder + when: + - cinder_csi_enabled + tags: + - cinder-csi-driver + - csi-driver + + - role: kubernetes-apps/csi_driver/aws_ebs + when: + - aws_ebs_csi_enabled + tags: + - aws-ebs-csi-driver + - csi-driver + + - role: kubernetes-apps/csi_driver/azuredisk + when: + - azure_csi_enabled + tags: + - azure-csi-driver + - csi-driver + + - role: kubernetes-apps/csi_driver/gcp_pd + when: + - gcp_pd_csi_enabled + tags: + - gcp-pd-csi-driver + - csi-driver + + - role: kubernetes-apps/csi_driver/upcloud + when: + - upcloud_csi_enabled + tags: + - upcloud-csi-driver + - csi-driver + + - role: kubernetes-apps/csi_driver/vsphere + when: + - vsphere_csi_enabled + tags: + - vsphere-csi-driver + - csi-driver + + - role: kubernetes-apps/persistent_volumes + when: + - persistent_volumes_enabled + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - persistent_volumes + + - role: kubernetes-apps/snapshots + when: inventory_hostname == groups['kube_control_plane'][0] + tags: + - snapshots + - csi-driver + + - role: kubernetes-apps/container_runtimes + when: + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - container-runtimes + + - role: kubernetes-apps/container_engine_accelerator + when: nvidia_accelerator_enabled + tags: + - container_engine_accelerator + + - role: kubernetes-apps/cloud_controller/oci + when: + - cloud_provider is defined + - cloud_provider == "oci" + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - oci + + - role: kubernetes-apps/metallb + when: + - metallb_enabled + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - metallb + + - role: kubernetes-apps/argocd + when: + - argocd_enabled + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - argocd diff --git a/kubespray/roles/kubernetes-apps/metallb/OWNERS b/kubespray/roles/kubernetes-apps/metallb/OWNERS new file mode 100644 index 0000000..b64c7bc --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metallb/OWNERS @@ -0,0 +1,5 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +approvers: +reviewers: + - oomichi diff --git a/kubespray/roles/kubernetes-apps/metallb/defaults/main.yml b/kubespray/roles/kubernetes-apps/metallb/defaults/main.yml new file mode 100644 index 0000000..e9012dc --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metallb/defaults/main.yml @@ -0,0 +1,18 @@ +--- +metallb_enabled: false +metallb_log_level: info +metallb_port: "7472" +metallb_memberlist_port: "7946" +metallb_speaker_enabled: "{{ metallb_enabled }}" +metallb_speaker_nodeselector: + kubernetes.io/os: "linux" +metallb_controller_nodeselector: + kubernetes.io/os: "linux" +metallb_speaker_tolerations: + - effect: NoSchedule + key: node-role.kubernetes.io/master + operator: Exists + - effect: NoSchedule + key: node-role.kubernetes.io/control-plane + operator: Exists +metallb_controller_tolerations: [] diff --git a/kubespray/roles/kubernetes-apps/metallb/tasks/main.yml b/kubespray/roles/kubernetes-apps/metallb/tasks/main.yml new file mode 100644 index 0000000..2988683 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metallb/tasks/main.yml @@ -0,0 +1,123 @@ +--- +- name: Kubernetes Apps | Check cluster settings for MetalLB + fail: + msg: "MetalLB require kube_proxy_strict_arp = true, see https://github.com/danderson/metallb/issues/153#issuecomment-518651132" + when: + - "kube_proxy_mode == 'ipvs' and not kube_proxy_strict_arp" + +- name: Kubernetes Apps | Check that the deprecated 'matallb_auto_assign' variable is not used anymore + fail: + msg: "'matallb_auto_assign' configuration variable is deprecated, please use 'metallb_auto_assign' instead" + when: + - matallb_auto_assign is defined + +- name: Kubernetes Apps | Check AppArmor status + command: which apparmor_parser + register: apparmor_status + when: + - podsecuritypolicy_enabled + - inventory_hostname == groups['kube_control_plane'][0] + failed_when: false + +- name: Kubernetes Apps | Set apparmor_enabled + set_fact: + apparmor_enabled: "{{ apparmor_status.rc == 0 }}" + when: + - podsecuritypolicy_enabled + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Apps | Lay Down MetalLB + become: true + template: + src: "metallb.yaml.j2" + dest: "{{ kube_config_dir }}/metallb.yaml" + mode: 0644 + register: metallb_rendering + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Apps | Install and configure MetalLB + kube: + name: "MetalLB" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/metallb.yaml" + state: "{{ metallb_rendering.changed | ternary('latest', 'present') }}" + wait: true + become: true + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Apps | Wait for MetalLB controller to be running + command: "{{ bin_dir }}/kubectl -n metallb-system wait --for=condition=ready pod -l app=metallb,component=controller --timeout=2m" + become: true + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: MetalLB | Address pools + when: + - inventory_hostname == groups['kube_control_plane'][0] + - metallb_config.address_pools is defined + block: + - name: MetalLB | Layout address pools template + ansible.builtin.template: + src: pools.yaml.j2 + dest: "{{ kube_config_dir }}/pools.yaml" + mode: 0644 + register: pools_rendering + + - name: MetalLB | Create address pools configuration + kube: + name: "MetalLB" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/pools.yaml" + state: "{{ pools_rendering.changed | ternary('latest', 'present') }}" + become: true + +- name: MetalLB | Layer2 + when: + - inventory_hostname == groups['kube_control_plane'][0] + - metallb_config.layer2 is defined + block: + - name: MetalLB | Layout layer2 template + ansible.builtin.template: + src: layer2.yaml.j2 + dest: "{{ kube_config_dir }}/layer2.yaml" + mode: 0644 + register: layer2_rendering + + - name: MetalLB | Create layer2 configuration + kube: + name: "MetalLB" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/layer2.yaml" + state: "{{ layer2_rendering.changed | ternary('latest', 'present') }}" + become: true + +- name: MetalLB | Layer3 + when: + - inventory_hostname == groups['kube_control_plane'][0] + - metallb_config.layer3 is defined + block: + - name: MetalLB | Layout layer3 template + ansible.builtin.template: + src: layer3.yaml.j2 + dest: "{{ kube_config_dir }}/layer3.yaml" + mode: 0644 + register: layer3_rendering + + - name: MetalLB | Create layer3 configuration + kube: + name: "MetalLB" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/layer3.yaml" + state: "{{ layer3_rendering.changed | ternary('latest', 'present') }}" + become: true + + +- name: Kubernetes Apps | Delete MetalLB ConfigMap + kube: + name: config + kubectl: "{{ bin_dir }}/kubectl" + resource: ConfigMap + namespace: metallb-system + state: absent diff --git a/kubespray/roles/kubernetes-apps/metallb/templates/layer2.yaml.j2 b/kubespray/roles/kubernetes-apps/metallb/templates/layer2.yaml.j2 new file mode 100644 index 0000000..d249732 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metallb/templates/layer2.yaml.j2 @@ -0,0 +1,19 @@ +#jinja2: trim_blocks: True, lstrip_blocks: True +# yamllint disable-file +--- + +# Create layer2 configuration +{% for entry in metallb_config.layer2 %} + +--- +# L2 Configuration +apiVersion: metallb.io/v1beta1 +kind: L2Advertisement +metadata: + name: "{{ entry }}" + namespace: metallb-system +spec: + ipAddressPools: + - "{{ entry }}" + +{% endfor %} diff --git a/kubespray/roles/kubernetes-apps/metallb/templates/layer3.yaml.j2 b/kubespray/roles/kubernetes-apps/metallb/templates/layer3.yaml.j2 new file mode 100644 index 0000000..490bae2 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metallb/templates/layer3.yaml.j2 @@ -0,0 +1,125 @@ +#jinja2: trim_blocks: True, lstrip_blocks: True +# yamllint disable-file +--- +# Create layer3 configuration +{% if metallb_config.layer3.communities is defined %} +{% for community_name, community in metallb_config.layer3.communities.items() %} +--- +apiVersion: metallb.io/v1beta1 +kind: Community +metadata: + name: "{{ community_name }}" + namespace: metallb-system +spec: + communities: + - name: "{{ community_name }}" + value: "{{ community }}" +{% endfor %} +{% endif %} +--- +apiVersion: metallb.io/v1beta1 +kind: Community +metadata: + name: well-known + namespace: metallb-system +spec: + communities: + - name: no-export + value: 65535:65281 + - name: no-advertise + value: 65535:65282 + - name: local-as + value: 65535:65283 + - name: nopeer + value: 65535:65284 + +# BGPAdvertisement is used to advertise address pools to the BGP peer. Specific pools can be listed to be advertised. +# Local BGP Advertisement specifies that the IP specified in the address pool will be used as remote source address for traffic entering your cluster from the remote peer. +# When using this option, be sure to use a subnet and routable IP for your address pool. +# This is good: 10.0.0.10/24. This is also good: 10.0.0.129/25. This is bad: 10.0.0.0/24. This is also bad: 10.0.0.128/25. +# In this example, 10.0.0.10 will be used as the remote source address. +# This is also bad: 10.0.0.10-10.0.0.25. Remember: you are working with aggregationLength, which specifies a subnet, not an IP range! +# The no-advertise community is set on the local advertisement to prevent this route from being published to the BGP peer. +# Your aggregationLength ideally is the same size as your address pool. + +{% for peer_name, peer in metallb_config.layer3.metallb_peers.items() %} + +{% if peer.aggregation_length is defined and peer.aggregation_length <= 30 %} + +--- +apiVersion: metallb.io/v1beta1 +kind: BGPAdvertisement +metadata: + name: "{{ peer_name }}-local" + namespace: metallb-system +spec: + aggregationLength: 32 + aggregationLengthV6: 128 + communities: + - no-advertise + localpref: "{{ peer.localpref | default("100") }}" + ipAddressPools: + {% for address_pool in peer.address_pool %} + - "{{ address_pool }}" + {% endfor %} +{% endif %} + +# External BGP Advertisement. The IP range specied in the address pool is advertised to the BGP peer. +--- +apiVersion: metallb.io/v1beta1 +kind: BGPAdvertisement +metadata: + name: "{{ peer_name }}-external" + namespace: metallb-system +spec: + {% if peer.aggregation_length is defined and peer.aggregation_length <= 30 %} + aggregationLength: {{ peer.aggregation_length }} + {% endif %} + ipAddressPools: + {% for address_pool in peer.address_pool %} + - "{{ address_pool }}" + {% endfor %} + {% if peer.communities is defined %} + {% for community in peer.communities %} + communities: + - "{{ community }}" + {% endfor %} + {% endif %} + + +# Configuration for the BGP peer. +--- +apiVersion: metallb.io/v1beta2 +kind: BGPPeer +metadata: + name: "{{ peer_name }}" + namespace: metallb-system +spec: + myASN: {{ peer.my_asn }} + peerASN: {{ peer.peer_asn }} + peerAddress: {{ peer.peer_address }} + {% if peer.peer_port is defined %} + peerPort: {{ peer.peer_port }} + {% else %} + peerPort: {{ metallb_config.layer3.defaults.peer_port }} + {% endif -%} + + {% if peer.password is defined %} + password: "{{ peer.password }}" + {% endif -%} + + {% if peer.router_id is defined %} + routerID: "{{ peer.router_id }}" + {% endif -%} + + {% if peer.hold_time is defined %} + holdTime: {{ peer.hold_time }} + {% elif metallb_config.layer3.defaults.hold_time is defined %} + holdTime: {{ metallb_config.layer3.defaults.hold_time }} + {% endif -%} + + {% if peer.multihop is defined %} + ebgpMultiHop: {{ peer.multihop }} + {% endif -%} + +{% endfor %} diff --git a/kubespray/roles/kubernetes-apps/metallb/templates/metallb.yaml.j2 b/kubespray/roles/kubernetes-apps/metallb/templates/metallb.yaml.j2 new file mode 100644 index 0000000..eab386f --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metallb/templates/metallb.yaml.j2 @@ -0,0 +1,2035 @@ +--- +apiVersion: v1 +kind: Namespace +metadata: + labels: + pod-security.kubernetes.io/audit: privileged + pod-security.kubernetes.io/enforce: privileged + pod-security.kubernetes.io/warn: privileged + name: metallb-system + +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.11.1 + name: addresspools.metallb.io +spec: + conversion: + strategy: Webhook + webhook: + clientConfig: + caBundle: LS0tLS1CRUdJTiBDRVJUSUZJQ0FURS0tLS0tDQpNSUlGWlRDQ0EwMmdBd0lCQWdJVU5GRW1XcTM3MVpKdGkrMmlSQzk1WmpBV1MxZ3dEUVlKS29aSWh2Y05BUUVMDQpCUUF3UWpFTE1Ba0dBMVVFQmhNQ1dGZ3hGVEFUQmdOVkJBY01ERVJsWm1GMWJIUWdRMmwwZVRFY01Cb0dBMVVFDQpDZ3dUUkdWbVlYVnNkQ0JEYjIxd1lXNTVJRXgwWkRBZUZ3MHlNakEzTVRrd09UTXlNek5hRncweU1qQTRNVGd3DQpPVE15TXpOYU1FSXhDekFKQmdOVkJBWVRBbGhZTVJVd0V3WURWUVFIREF4RVpXWmhkV3gwSUVOcGRIa3hIREFhDQpCZ05WQkFvTUUwUmxabUYxYkhRZ1EyOXRjR0Z1ZVNCTWRHUXdnZ0lpTUEwR0NTcUdTSWIzRFFFQkFRVUFBNElDDQpEd0F3Z2dJS0FvSUNBUUNxVFpxMWZRcC9vYkdlenhES0o3OVB3Ny94azJwellualNzMlkzb1ZYSm5sRmM4YjVlDQpma2ZZQnY2bndscW1keW5PL2phWFBaQmRQSS82aFdOUDBkdVhadEtWU0NCUUpyZzEyOGNXb3F0MGNTN3pLb1VpDQpvcU1tQ0QvRXVBeFFNZjhRZDF2c1gvVllkZ0poVTZBRXJLZEpIaXpFOUJtUkNkTDBGMW1OVW55Rk82UnRtWFZUDQpidkxsTDVYeTc2R0FaQVBLOFB4aVlDa0NtbDdxN0VnTWNiOXlLWldCYmlxQ3VkTXE5TGJLNmdKNzF6YkZnSXV4DQo1L1pXK2JraTB2RlplWk9ZODUxb1psckFUNzJvMDI4NHNTWW9uN0pHZVZkY3NoUnh5R1VpSFpSTzdkaXZVTDVTDQpmM2JmSDFYbWY1ZDQzT0NWTWRuUUV2NWVaOG8zeWVLa3ZrbkZQUGVJMU9BbjdGbDlFRVNNR2dhOGFaSG1URSttDQpsLzlMSmdDYjBnQmtPT0M0WnV4bWh2aERKV1EzWnJCS3pMQlNUZXN0NWlLNVlwcXRWVVk2THRyRW9FelVTK1lsDQpwWndXY2VQWHlHeHM5ZURsR3lNVmQraW15Y3NTU1UvVno2Mmx6MnZCS21NTXBkYldDQWhud0RsRTVqU2dyMjRRDQp0eGNXLys2N3d5KzhuQlI3UXdqVTFITndVRjBzeERWdEwrZ1NHVERnSEVZSlhZelYvT05zMy94TkpoVFNPSkxNDQpoeXNVdyttaGdackdhbUdXcHVIVU1DUitvTWJzMTc1UkcrQjJnUFFHVytPTjJnUTRyOXN2b0ZBNHBBQm8xd1dLDQpRYjRhY3pmeVVscElBOVFoSmFsZEY3S3dPSHVlV3gwRUNrNXg0T2tvVDBvWVp0dzFiR0JjRGtaSmF3SURBUUFCDQpvMU13VVRBZEJnTlZIUTRFRmdRVW90UlNIUm9IWTEyRFZ4R0NCdEhpb1g2ZmVFQXdId1lEVlIwakJCZ3dGb0FVDQpvdFJTSFJvSFkxMkRWeEdDQnRIaW9YNmZlRUF3RHdZRFZSMFRBUUgvQkFVd0F3RUIvekFOQmdrcWhraUc5dzBCDQpBUXNGQUFPQ0FnRUFSbkpsWWRjMTFHd0VxWnh6RDF2R3BDR2pDN2VWTlQ3aVY1d3IybXlybHdPYi9aUWFEa0xYDQpvVStaOVVXT1VlSXJTdzUydDdmQUpvVVAwSm5iYkMveVIrU1lqUGhvUXNiVHduOTc2ZldBWTduM3FMOXhCd1Y0DQphek41OXNjeUp0dlhMeUtOL2N5ak1ReDRLajBIMFg0bWJ6bzVZNUtzWWtYVU0vOEFPdWZMcEd0S1NGVGgrSEFDDQpab1Q5YnZHS25adnNHd0tYZFF0Wnh0akhaUjVqK3U3ZGtQOTJBT051RFNabS8rWVV4b2tBK09JbzdSR3BwSHNXDQo1ZTdNY0FTVXRtb1FORXd6dVFoVkJaRWQ1OGtKYjUrV0VWbGNzanlXNnRTbzErZ25tTWNqR1BsMWgxR2hVbjV4DQpFY0lWRnBIWXM5YWo1NmpBSjk1MVQvZjhMaWxmTlVnanBLQ0c1bnl0SUt3emxhOHNtdGlPdm1UNEpYbXBwSkI2DQo4bmdHRVluVjUrUTYwWFJ2OEhSSGp1VG9CRHVhaERrVDA2R1JGODU1d09FR2V4bkZpMXZYWUxLVllWb1V2MXRKDQo4dVdUR1pwNllDSVJldlBqbzg5ZytWTlJSaVFYUThJd0dybXE5c0RoVTlqTjA0SjdVL1RvRDFpNHE3VnlsRUc5DQorV1VGNkNLaEdBeTJIaEhwVncyTGFoOS9lUzdZMUZ1YURrWmhPZG1laG1BOCtqdHNZamJadnR5Mm1SWlF0UUZzDQpUU1VUUjREbUR2bVVPRVRmeStpRHdzK2RkWXVNTnJGeVVYV2dkMnpBQU4ydVl1UHFGY2pRcFNPODFzVTJTU3R3DQoxVzAyeUtYOGJEYmZFdjBzbUh3UzliQnFlSGo5NEM1Mjg0YXpsdTBmaUdpTm1OUEM4ckJLRmhBPQ0KLS0tLS1FTkQgQ0VSVElGSUNBVEUtLS0tLQ== + service: + name: webhook-service + namespace: metallb-system + path: /convert + conversionReviewVersions: + - v1alpha1 + - v1beta1 + group: metallb.io + names: + kind: AddressPool + listKind: AddressPoolList + plural: addresspools + singular: addresspool + scope: Namespaced + versions: + - deprecated: true + deprecationWarning: metallb.io v1alpha1 AddressPool is deprecated + name: v1alpha1 + schema: + openAPIV3Schema: + description: AddressPool is the Schema for the addresspools API. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: AddressPoolSpec defines the desired state of AddressPool. + properties: + addresses: + description: A list of IP address ranges over which MetalLB has authority. + You can list multiple ranges in a single pool, they will all share + the same settings. Each range can be either a CIDR prefix, or an + explicit start-end range of IPs. + items: + type: string + type: array + autoAssign: + default: true + description: AutoAssign flag used to prevent MetallB from automatic + allocation for a pool. + type: boolean + bgpAdvertisements: + description: When an IP is allocated from this pool, how should it + be translated into BGP announcements? + items: + properties: + aggregationLength: + default: 32 + description: The aggregation-length advertisement option lets + you “roll up” the /32s into a larger prefix. + format: int32 + minimum: 1 + type: integer + aggregationLengthV6: + default: 128 + description: Optional, defaults to 128 (i.e. no aggregation) + if not specified. + format: int32 + type: integer + communities: + description: BGP communities + items: + type: string + type: array + localPref: + description: BGP LOCAL_PREF attribute which is used by BGP best + path algorithm, Path with higher localpref is preferred over + one with lower localpref. + format: int32 + type: integer + type: object + type: array + protocol: + description: Protocol can be used to select how the announcement is + done. + enum: + - layer2 + - bgp + type: string + required: + - addresses + - protocol + type: object + status: + description: AddressPoolStatus defines the observed state of AddressPool. + type: object + required: + - spec + type: object + served: true + storage: false + subresources: + status: {} + - deprecated: true + deprecationWarning: metallb.io v1beta1 AddressPool is deprecated, consider using + IPAddressPool + name: v1beta1 + schema: + openAPIV3Schema: + description: AddressPool represents a pool of IP addresses that can be allocated + to LoadBalancer services. AddressPool is deprecated and being replaced by + IPAddressPool. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: AddressPoolSpec defines the desired state of AddressPool. + properties: + addresses: + description: A list of IP address ranges over which MetalLB has authority. + You can list multiple ranges in a single pool, they will all share + the same settings. Each range can be either a CIDR prefix, or an + explicit start-end range of IPs. + items: + type: string + type: array + autoAssign: + default: true + description: AutoAssign flag used to prevent MetallB from automatic + allocation for a pool. + type: boolean + bgpAdvertisements: + description: Drives how an IP allocated from this pool should translated + into BGP announcements. + items: + properties: + aggregationLength: + default: 32 + description: The aggregation-length advertisement option lets + you “roll up” the /32s into a larger prefix. + format: int32 + minimum: 1 + type: integer + aggregationLengthV6: + default: 128 + description: Optional, defaults to 128 (i.e. no aggregation) + if not specified. + format: int32 + type: integer + communities: + description: BGP communities to be associated with the given + advertisement. + items: + type: string + type: array + localPref: + description: BGP LOCAL_PREF attribute which is used by BGP best + path algorithm, Path with higher localpref is preferred over + one with lower localpref. + format: int32 + type: integer + type: object + type: array + protocol: + description: Protocol can be used to select how the announcement is + done. + enum: + - layer2 + - bgp + type: string + required: + - addresses + - protocol + type: object + status: + description: AddressPoolStatus defines the observed state of AddressPool. + type: object + required: + - spec + type: object + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.11.1 + creationTimestamp: null + name: bfdprofiles.metallb.io +spec: + group: metallb.io + names: + kind: BFDProfile + listKind: BFDProfileList + plural: bfdprofiles + singular: bfdprofile + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.passiveMode + name: Passive Mode + type: boolean + - jsonPath: .spec.transmitInterval + name: Transmit Interval + type: integer + - jsonPath: .spec.receiveInterval + name: Receive Interval + type: integer + - jsonPath: .spec.detectMultiplier + name: Multiplier + type: integer + name: v1beta1 + schema: + openAPIV3Schema: + description: BFDProfile represents the settings of the bfd session that can + be optionally associated with a BGP session. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: BFDProfileSpec defines the desired state of BFDProfile. + properties: + detectMultiplier: + description: Configures the detection multiplier to determine packet + loss. The remote transmission interval will be multiplied by this + value to determine the connection loss detection timer. + format: int32 + maximum: 255 + minimum: 2 + type: integer + echoInterval: + description: Configures the minimal echo receive transmission interval + that this system is capable of handling in milliseconds. Defaults + to 50ms + format: int32 + maximum: 60000 + minimum: 10 + type: integer + echoMode: + description: Enables or disables the echo transmission mode. This + mode is disabled by default, and not supported on multi hops setups. + type: boolean + minimumTtl: + description: 'For multi hop sessions only: configure the minimum expected + TTL for an incoming BFD control packet.' + format: int32 + maximum: 254 + minimum: 1 + type: integer + passiveMode: + description: 'Mark session as passive: a passive session will not + attempt to start the connection and will wait for control packets + from peer before it begins replying.' + type: boolean + receiveInterval: + description: The minimum interval that this system is capable of receiving + control packets in milliseconds. Defaults to 300ms. + format: int32 + maximum: 60000 + minimum: 10 + type: integer + transmitInterval: + description: The minimum transmission interval (less jitter) that + this system wants to use to send BFD control packets in milliseconds. + Defaults to 300ms + format: int32 + maximum: 60000 + minimum: 10 + type: integer + type: object + status: + description: BFDProfileStatus defines the observed state of BFDProfile. + type: object + type: object + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.11.1 + creationTimestamp: null + name: bgpadvertisements.metallb.io +spec: + group: metallb.io + names: + kind: BGPAdvertisement + listKind: BGPAdvertisementList + plural: bgpadvertisements + singular: bgpadvertisement + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.ipAddressPools + name: IPAddressPools + type: string + - jsonPath: .spec.ipAddressPoolSelectors + name: IPAddressPool Selectors + type: string + - jsonPath: .spec.peers + name: Peers + type: string + - jsonPath: .spec.nodeSelectors + name: Node Selectors + priority: 10 + type: string + name: v1beta1 + schema: + openAPIV3Schema: + description: BGPAdvertisement allows to advertise the IPs coming from the + selected IPAddressPools via BGP, setting the parameters of the BGP Advertisement. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: BGPAdvertisementSpec defines the desired state of BGPAdvertisement. + properties: + aggregationLength: + default: 32 + description: The aggregation-length advertisement option lets you + “roll up” the /32s into a larger prefix. Defaults to 32. Works for + IPv4 addresses. + format: int32 + minimum: 1 + type: integer + aggregationLengthV6: + default: 128 + description: The aggregation-length advertisement option lets you + “roll up” the /128s into a larger prefix. Defaults to 128. Works + for IPv6 addresses. + format: int32 + type: integer + communities: + description: The BGP communities to be associated with the announcement. + Each item can be a community of the form 1234:1234 or the name of + an alias defined in the Community CRD. + items: + type: string + type: array + ipAddressPoolSelectors: + description: A selector for the IPAddressPools which would get advertised + via this advertisement. If no IPAddressPool is selected by this + or by the list, the advertisement is applied to all the IPAddressPools. + items: + description: A label selector is a label query over a set of resources. + The result of matchLabels and matchExpressions are ANDed. An empty + label selector matches all objects. A null label selector matches + no objects. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. + The requirements are ANDed. + items: + description: A label selector requirement is a selector that + contains values, a key, and an operator that relates the + key and values. + properties: + key: + description: key is the label key that the selector applies + to. + type: string + operator: + description: operator represents a key's relationship + to a set of values. Valid operators are In, NotIn, Exists + and DoesNotExist. + type: string + values: + description: values is an array of string values. If the + operator is In or NotIn, the values array must be non-empty. + If the operator is Exists or DoesNotExist, the values + array must be empty. This array is replaced during a + strategic merge patch. + items: + type: string + type: array + required: + - key + - operator + type: object + type: array + matchLabels: + additionalProperties: + type: string + description: matchLabels is a map of {key,value} pairs. A single + {key,value} in the matchLabels map is equivalent to an element + of matchExpressions, whose key field is "key", the operator + is "In", and the values array contains only "value". The requirements + are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + type: array + ipAddressPools: + description: The list of IPAddressPools to advertise via this advertisement, + selected by name. + items: + type: string + type: array + localPref: + description: The BGP LOCAL_PREF attribute which is used by BGP best + path algorithm, Path with higher localpref is preferred over one + with lower localpref. + format: int32 + type: integer + nodeSelectors: + description: NodeSelectors allows to limit the nodes to announce as + next hops for the LoadBalancer IP. When empty, all the nodes having are + announced as next hops. + items: + description: A label selector is a label query over a set of resources. + The result of matchLabels and matchExpressions are ANDed. An empty + label selector matches all objects. A null label selector matches + no objects. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. + The requirements are ANDed. + items: + description: A label selector requirement is a selector that + contains values, a key, and an operator that relates the + key and values. + properties: + key: + description: key is the label key that the selector applies + to. + type: string + operator: + description: operator represents a key's relationship + to a set of values. Valid operators are In, NotIn, Exists + and DoesNotExist. + type: string + values: + description: values is an array of string values. If the + operator is In or NotIn, the values array must be non-empty. + If the operator is Exists or DoesNotExist, the values + array must be empty. This array is replaced during a + strategic merge patch. + items: + type: string + type: array + required: + - key + - operator + type: object + type: array + matchLabels: + additionalProperties: + type: string + description: matchLabels is a map of {key,value} pairs. A single + {key,value} in the matchLabels map is equivalent to an element + of matchExpressions, whose key field is "key", the operator + is "In", and the values array contains only "value". The requirements + are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + type: array + peers: + description: Peers limits the bgppeer to advertise the ips of the + selected pools to. When empty, the loadbalancer IP is announced + to all the BGPPeers configured. + items: + type: string + type: array + type: object + status: + description: BGPAdvertisementStatus defines the observed state of BGPAdvertisement. + type: object + type: object + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.11.1 + name: bgppeers.metallb.io +spec: + conversion: + strategy: Webhook + webhook: + clientConfig: + caBundle: LS0tLS1CRUdJTiBDRVJUSUZJQ0FURS0tLS0tDQpNSUlGWlRDQ0EwMmdBd0lCQWdJVU5GRW1XcTM3MVpKdGkrMmlSQzk1WmpBV1MxZ3dEUVlKS29aSWh2Y05BUUVMDQpCUUF3UWpFTE1Ba0dBMVVFQmhNQ1dGZ3hGVEFUQmdOVkJBY01ERVJsWm1GMWJIUWdRMmwwZVRFY01Cb0dBMVVFDQpDZ3dUUkdWbVlYVnNkQ0JEYjIxd1lXNTVJRXgwWkRBZUZ3MHlNakEzTVRrd09UTXlNek5hRncweU1qQTRNVGd3DQpPVE15TXpOYU1FSXhDekFKQmdOVkJBWVRBbGhZTVJVd0V3WURWUVFIREF4RVpXWmhkV3gwSUVOcGRIa3hIREFhDQpCZ05WQkFvTUUwUmxabUYxYkhRZ1EyOXRjR0Z1ZVNCTWRHUXdnZ0lpTUEwR0NTcUdTSWIzRFFFQkFRVUFBNElDDQpEd0F3Z2dJS0FvSUNBUUNxVFpxMWZRcC9vYkdlenhES0o3OVB3Ny94azJwellualNzMlkzb1ZYSm5sRmM4YjVlDQpma2ZZQnY2bndscW1keW5PL2phWFBaQmRQSS82aFdOUDBkdVhadEtWU0NCUUpyZzEyOGNXb3F0MGNTN3pLb1VpDQpvcU1tQ0QvRXVBeFFNZjhRZDF2c1gvVllkZ0poVTZBRXJLZEpIaXpFOUJtUkNkTDBGMW1OVW55Rk82UnRtWFZUDQpidkxsTDVYeTc2R0FaQVBLOFB4aVlDa0NtbDdxN0VnTWNiOXlLWldCYmlxQ3VkTXE5TGJLNmdKNzF6YkZnSXV4DQo1L1pXK2JraTB2RlplWk9ZODUxb1psckFUNzJvMDI4NHNTWW9uN0pHZVZkY3NoUnh5R1VpSFpSTzdkaXZVTDVTDQpmM2JmSDFYbWY1ZDQzT0NWTWRuUUV2NWVaOG8zeWVLa3ZrbkZQUGVJMU9BbjdGbDlFRVNNR2dhOGFaSG1URSttDQpsLzlMSmdDYjBnQmtPT0M0WnV4bWh2aERKV1EzWnJCS3pMQlNUZXN0NWlLNVlwcXRWVVk2THRyRW9FelVTK1lsDQpwWndXY2VQWHlHeHM5ZURsR3lNVmQraW15Y3NTU1UvVno2Mmx6MnZCS21NTXBkYldDQWhud0RsRTVqU2dyMjRRDQp0eGNXLys2N3d5KzhuQlI3UXdqVTFITndVRjBzeERWdEwrZ1NHVERnSEVZSlhZelYvT05zMy94TkpoVFNPSkxNDQpoeXNVdyttaGdackdhbUdXcHVIVU1DUitvTWJzMTc1UkcrQjJnUFFHVytPTjJnUTRyOXN2b0ZBNHBBQm8xd1dLDQpRYjRhY3pmeVVscElBOVFoSmFsZEY3S3dPSHVlV3gwRUNrNXg0T2tvVDBvWVp0dzFiR0JjRGtaSmF3SURBUUFCDQpvMU13VVRBZEJnTlZIUTRFRmdRVW90UlNIUm9IWTEyRFZ4R0NCdEhpb1g2ZmVFQXdId1lEVlIwakJCZ3dGb0FVDQpvdFJTSFJvSFkxMkRWeEdDQnRIaW9YNmZlRUF3RHdZRFZSMFRBUUgvQkFVd0F3RUIvekFOQmdrcWhraUc5dzBCDQpBUXNGQUFPQ0FnRUFSbkpsWWRjMTFHd0VxWnh6RDF2R3BDR2pDN2VWTlQ3aVY1d3IybXlybHdPYi9aUWFEa0xYDQpvVStaOVVXT1VlSXJTdzUydDdmQUpvVVAwSm5iYkMveVIrU1lqUGhvUXNiVHduOTc2ZldBWTduM3FMOXhCd1Y0DQphek41OXNjeUp0dlhMeUtOL2N5ak1ReDRLajBIMFg0bWJ6bzVZNUtzWWtYVU0vOEFPdWZMcEd0S1NGVGgrSEFDDQpab1Q5YnZHS25adnNHd0tYZFF0Wnh0akhaUjVqK3U3ZGtQOTJBT051RFNabS8rWVV4b2tBK09JbzdSR3BwSHNXDQo1ZTdNY0FTVXRtb1FORXd6dVFoVkJaRWQ1OGtKYjUrV0VWbGNzanlXNnRTbzErZ25tTWNqR1BsMWgxR2hVbjV4DQpFY0lWRnBIWXM5YWo1NmpBSjk1MVQvZjhMaWxmTlVnanBLQ0c1bnl0SUt3emxhOHNtdGlPdm1UNEpYbXBwSkI2DQo4bmdHRVluVjUrUTYwWFJ2OEhSSGp1VG9CRHVhaERrVDA2R1JGODU1d09FR2V4bkZpMXZYWUxLVllWb1V2MXRKDQo4dVdUR1pwNllDSVJldlBqbzg5ZytWTlJSaVFYUThJd0dybXE5c0RoVTlqTjA0SjdVL1RvRDFpNHE3VnlsRUc5DQorV1VGNkNLaEdBeTJIaEhwVncyTGFoOS9lUzdZMUZ1YURrWmhPZG1laG1BOCtqdHNZamJadnR5Mm1SWlF0UUZzDQpUU1VUUjREbUR2bVVPRVRmeStpRHdzK2RkWXVNTnJGeVVYV2dkMnpBQU4ydVl1UHFGY2pRcFNPODFzVTJTU3R3DQoxVzAyeUtYOGJEYmZFdjBzbUh3UzliQnFlSGo5NEM1Mjg0YXpsdTBmaUdpTm1OUEM4ckJLRmhBPQ0KLS0tLS1FTkQgQ0VSVElGSUNBVEUtLS0tLQ== + service: + name: webhook-service + namespace: metallb-system + path: /convert + conversionReviewVersions: + - v1beta1 + - v1beta2 + group: metallb.io + names: + kind: BGPPeer + listKind: BGPPeerList + plural: bgppeers + singular: bgppeer + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.peerAddress + name: Address + type: string + - jsonPath: .spec.peerASN + name: ASN + type: string + - jsonPath: .spec.bfdProfile + name: BFD Profile + type: string + - jsonPath: .spec.ebgpMultiHop + name: Multi Hops + type: string + name: v1beta1 + schema: + openAPIV3Schema: + description: BGPPeer is the Schema for the peers API. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: BGPPeerSpec defines the desired state of Peer. + properties: + bfdProfile: + type: string + ebgpMultiHop: + description: EBGP peer is multi-hops away + type: boolean + holdTime: + description: Requested BGP hold time, per RFC4271. + type: string + keepaliveTime: + description: Requested BGP keepalive time, per RFC4271. + type: string + myASN: + description: AS number to use for the local end of the session. + format: int32 + maximum: 4294967295 + minimum: 0 + type: integer + nodeSelectors: + description: Only connect to this peer on nodes that match one of + these selectors. + items: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + minItems: 1 + type: array + required: + - key + - operator + - values + type: object + type: array + matchLabels: + additionalProperties: + type: string + type: object + type: object + type: array + password: + description: Authentication password for routers enforcing TCP MD5 + authenticated sessions + type: string + peerASN: + description: AS number to expect from the remote end of the session. + format: int32 + maximum: 4294967295 + minimum: 0 + type: integer + peerAddress: + description: Address to dial when establishing the session. + type: string + peerPort: + description: Port to dial when establishing the session. + maximum: 16384 + minimum: 0 + type: integer + routerID: + description: BGP router ID to advertise to the peer + type: string + sourceAddress: + description: Source address to use when establishing the session. + type: string + required: + - myASN + - peerASN + - peerAddress + type: object + status: + description: BGPPeerStatus defines the observed state of Peer. + type: object + type: object + served: true + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .spec.peerAddress + name: Address + type: string + - jsonPath: .spec.peerASN + name: ASN + type: string + - jsonPath: .spec.bfdProfile + name: BFD Profile + type: string + - jsonPath: .spec.ebgpMultiHop + name: Multi Hops + type: string + name: v1beta2 + schema: + openAPIV3Schema: + description: BGPPeer is the Schema for the peers API. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: BGPPeerSpec defines the desired state of Peer. + properties: + bfdProfile: + description: The name of the BFD Profile to be used for the BFD session + associated to the BGP session. If not set, the BFD session won't + be set up. + type: string + ebgpMultiHop: + description: To set if the BGPPeer is multi-hops away. Needed for + FRR mode only. + type: boolean + holdTime: + description: Requested BGP hold time, per RFC4271. + type: string + keepaliveTime: + description: Requested BGP keepalive time, per RFC4271. + type: string + myASN: + description: AS number to use for the local end of the session. + format: int32 + maximum: 4294967295 + minimum: 0 + type: integer + nodeSelectors: + description: Only connect to this peer on nodes that match one of + these selectors. + items: + description: A label selector is a label query over a set of resources. + The result of matchLabels and matchExpressions are ANDed. An empty + label selector matches all objects. A null label selector matches + no objects. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. + The requirements are ANDed. + items: + description: A label selector requirement is a selector that + contains values, a key, and an operator that relates the + key and values. + properties: + key: + description: key is the label key that the selector applies + to. + type: string + operator: + description: operator represents a key's relationship + to a set of values. Valid operators are In, NotIn, Exists + and DoesNotExist. + type: string + values: + description: values is an array of string values. If the + operator is In or NotIn, the values array must be non-empty. + If the operator is Exists or DoesNotExist, the values + array must be empty. This array is replaced during a + strategic merge patch. + items: + type: string + type: array + required: + - key + - operator + type: object + type: array + matchLabels: + additionalProperties: + type: string + description: matchLabels is a map of {key,value} pairs. A single + {key,value} in the matchLabels map is equivalent to an element + of matchExpressions, whose key field is "key", the operator + is "In", and the values array contains only "value". The requirements + are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + type: array + password: + description: Authentication password for routers enforcing TCP MD5 + authenticated sessions + type: string + passwordSecret: + description: passwordSecret is name of the authentication secret for + BGP Peer. the secret must be of type "kubernetes.io/basic-auth", + and created in the same namespace as the MetalLB deployment. The + password is stored in the secret as the key "password". + properties: + name: + description: name is unique within a namespace to reference a + secret resource. + type: string + namespace: + description: namespace defines the space within which the secret + name must be unique. + type: string + type: object + x-kubernetes-map-type: atomic + peerASN: + description: AS number to expect from the remote end of the session. + format: int32 + maximum: 4294967295 + minimum: 0 + type: integer + peerAddress: + description: Address to dial when establishing the session. + type: string + peerPort: + default: 179 + description: Port to dial when establishing the session. + maximum: 16384 + minimum: 0 + type: integer + routerID: + description: BGP router ID to advertise to the peer + type: string + sourceAddress: + description: Source address to use when establishing the session. + type: string + vrf: + description: To set if we want to peer with the BGPPeer using an interface + belonging to a host vrf + type: string + required: + - myASN + - peerASN + - peerAddress + type: object + status: + description: BGPPeerStatus defines the observed state of Peer. + type: object + type: object + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.11.1 + creationTimestamp: null + name: communities.metallb.io +spec: + group: metallb.io + names: + kind: Community + listKind: CommunityList + plural: communities + singular: community + scope: Namespaced + versions: + - name: v1beta1 + schema: + openAPIV3Schema: + description: Community is a collection of aliases for communities. Users can + define named aliases to be used in the BGPPeer CRD. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: CommunitySpec defines the desired state of Community. + properties: + communities: + items: + properties: + name: + description: The name of the alias for the community. + type: string + value: + description: The BGP community value corresponding to the given + name. + type: string + type: object + type: array + type: object + status: + description: CommunityStatus defines the observed state of Community. + type: object + type: object + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.11.1 + creationTimestamp: null + name: ipaddresspools.metallb.io +spec: + group: metallb.io + names: + kind: IPAddressPool + listKind: IPAddressPoolList + plural: ipaddresspools + singular: ipaddresspool + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.autoAssign + name: Auto Assign + type: boolean + - jsonPath: .spec.avoidBuggyIPs + name: Avoid Buggy IPs + type: boolean + - jsonPath: .spec.addresses + name: Addresses + type: string + name: v1beta1 + schema: + openAPIV3Schema: + description: IPAddressPool represents a pool of IP addresses that can be allocated + to LoadBalancer services. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: IPAddressPoolSpec defines the desired state of IPAddressPool. + properties: + addresses: + description: A list of IP address ranges over which MetalLB has authority. + You can list multiple ranges in a single pool, they will all share + the same settings. Each range can be either a CIDR prefix, or an + explicit start-end range of IPs. + items: + type: string + type: array + autoAssign: + default: true + description: AutoAssign flag used to prevent MetallB from automatic + allocation for a pool. + type: boolean + avoidBuggyIPs: + default: false + description: AvoidBuggyIPs prevents addresses ending with .0 and .255 + to be used by a pool. + type: boolean + serviceAllocation: + description: AllocateTo makes ip pool allocation to specific namespace + and/or service. The controller will use the pool with lowest value + of priority in case of multiple matches. A pool with no priority + set will be used only if the pools with priority can't be used. + If multiple matching IPAddressPools are available it will check + for the availability of IPs sorting the matching IPAddressPools + by priority, starting from the highest to the lowest. If multiple + IPAddressPools have the same priority, choice will be random. + properties: + namespaceSelectors: + description: NamespaceSelectors list of label selectors to select + namespace(s) for ip pool, an alternative to using namespace + list. + items: + description: A label selector is a label query over a set of + resources. The result of matchLabels and matchExpressions + are ANDed. An empty label selector matches all objects. A + null label selector matches no objects. + properties: + matchExpressions: + description: matchExpressions is a list of label selector + requirements. The requirements are ANDed. + items: + description: A label selector requirement is a selector + that contains values, a key, and an operator that relates + the key and values. + properties: + key: + description: key is the label key that the selector + applies to. + type: string + operator: + description: operator represents a key's relationship + to a set of values. Valid operators are In, NotIn, + Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. + If the operator is In or NotIn, the values array + must be non-empty. If the operator is Exists or + DoesNotExist, the values array must be empty. This + array is replaced during a strategic merge patch. + items: + type: string + type: array + required: + - key + - operator + type: object + type: array + matchLabels: + additionalProperties: + type: string + description: matchLabels is a map of {key,value} pairs. + A single {key,value} in the matchLabels map is equivalent + to an element of matchExpressions, whose key field is + "key", the operator is "In", and the values array contains + only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + type: array + namespaces: + description: Namespaces list of namespace(s) on which ip pool + can be attached. + items: + type: string + type: array + priority: + description: Priority priority given for ip pool while ip allocation + on a service. + type: integer + serviceSelectors: + description: ServiceSelectors list of label selector to select + service(s) for which ip pool can be used for ip allocation. + items: + description: A label selector is a label query over a set of + resources. The result of matchLabels and matchExpressions + are ANDed. An empty label selector matches all objects. A + null label selector matches no objects. + properties: + matchExpressions: + description: matchExpressions is a list of label selector + requirements. The requirements are ANDed. + items: + description: A label selector requirement is a selector + that contains values, a key, and an operator that relates + the key and values. + properties: + key: + description: key is the label key that the selector + applies to. + type: string + operator: + description: operator represents a key's relationship + to a set of values. Valid operators are In, NotIn, + Exists and DoesNotExist. + type: string + values: + description: values is an array of string values. + If the operator is In or NotIn, the values array + must be non-empty. If the operator is Exists or + DoesNotExist, the values array must be empty. This + array is replaced during a strategic merge patch. + items: + type: string + type: array + required: + - key + - operator + type: object + type: array + matchLabels: + additionalProperties: + type: string + description: matchLabels is a map of {key,value} pairs. + A single {key,value} in the matchLabels map is equivalent + to an element of matchExpressions, whose key field is + "key", the operator is "In", and the values array contains + only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + type: array + type: object + required: + - addresses + type: object + status: + description: IPAddressPoolStatus defines the observed state of IPAddressPool. + type: object + required: + - spec + type: object + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.11.1 + creationTimestamp: null + name: l2advertisements.metallb.io +spec: + group: metallb.io + names: + kind: L2Advertisement + listKind: L2AdvertisementList + plural: l2advertisements + singular: l2advertisement + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.ipAddressPools + name: IPAddressPools + type: string + - jsonPath: .spec.ipAddressPoolSelectors + name: IPAddressPool Selectors + type: string + - jsonPath: .spec.interfaces + name: Interfaces + type: string + - jsonPath: .spec.nodeSelectors + name: Node Selectors + priority: 10 + type: string + name: v1beta1 + schema: + openAPIV3Schema: + description: L2Advertisement allows to advertise the LoadBalancer IPs provided + by the selected pools via L2. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: L2AdvertisementSpec defines the desired state of L2Advertisement. + properties: + interfaces: + description: A list of interfaces to announce from. The LB IP will + be announced only from these interfaces. If the field is not set, + we advertise from all the interfaces on the host. + items: + type: string + type: array + ipAddressPoolSelectors: + description: A selector for the IPAddressPools which would get advertised + via this advertisement. If no IPAddressPool is selected by this + or by the list, the advertisement is applied to all the IPAddressPools. + items: + description: A label selector is a label query over a set of resources. + The result of matchLabels and matchExpressions are ANDed. An empty + label selector matches all objects. A null label selector matches + no objects. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. + The requirements are ANDed. + items: + description: A label selector requirement is a selector that + contains values, a key, and an operator that relates the + key and values. + properties: + key: + description: key is the label key that the selector applies + to. + type: string + operator: + description: operator represents a key's relationship + to a set of values. Valid operators are In, NotIn, Exists + and DoesNotExist. + type: string + values: + description: values is an array of string values. If the + operator is In or NotIn, the values array must be non-empty. + If the operator is Exists or DoesNotExist, the values + array must be empty. This array is replaced during a + strategic merge patch. + items: + type: string + type: array + required: + - key + - operator + type: object + type: array + matchLabels: + additionalProperties: + type: string + description: matchLabels is a map of {key,value} pairs. A single + {key,value} in the matchLabels map is equivalent to an element + of matchExpressions, whose key field is "key", the operator + is "In", and the values array contains only "value". The requirements + are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + type: array + ipAddressPools: + description: The list of IPAddressPools to advertise via this advertisement, + selected by name. + items: + type: string + type: array + nodeSelectors: + description: NodeSelectors allows to limit the nodes to announce as + next hops for the LoadBalancer IP. When empty, all the nodes having are + announced as next hops. + items: + description: A label selector is a label query over a set of resources. + The result of matchLabels and matchExpressions are ANDed. An empty + label selector matches all objects. A null label selector matches + no objects. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. + The requirements are ANDed. + items: + description: A label selector requirement is a selector that + contains values, a key, and an operator that relates the + key and values. + properties: + key: + description: key is the label key that the selector applies + to. + type: string + operator: + description: operator represents a key's relationship + to a set of values. Valid operators are In, NotIn, Exists + and DoesNotExist. + type: string + values: + description: values is an array of string values. If the + operator is In or NotIn, the values array must be non-empty. + If the operator is Exists or DoesNotExist, the values + array must be empty. This array is replaced during a + strategic merge patch. + items: + type: string + type: array + required: + - key + - operator + type: object + type: array + matchLabels: + additionalProperties: + type: string + description: matchLabels is a map of {key,value} pairs. A single + {key,value} in the matchLabels map is equivalent to an element + of matchExpressions, whose key field is "key", the operator + is "In", and the values array contains only "value". The requirements + are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + type: array + type: object + status: + description: L2AdvertisementStatus defines the observed state of L2Advertisement. + type: object + type: object + served: true + storage: true + subresources: + status: {} +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + app: metallb + pod-security.kubernetes.io/audit: privileged + pod-security.kubernetes.io/enforce: privileged + pod-security.kubernetes.io/warn: privileged + name: controller + namespace: metallb-system + +{% if metallb_speaker_enabled %} +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + app: metallb + name: speaker + namespace: metallb-system +{% endif %} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + labels: + app: metallb + name: controller + namespace: metallb-system +rules: +- apiGroups: + - "" + resources: + - secrets + verbs: + - create + - delete + - get + - list + - patch + - update + - watch +- apiGroups: + - "" + resourceNames: + - memberlist + resources: + - secrets + verbs: + - list +- apiGroups: + - apps + resourceNames: + - controller + resources: + - deployments + verbs: + - get +- apiGroups: + - metallb.io + resources: + - bgppeers + verbs: + - get + - list +- apiGroups: + - metallb.io + resources: + - addresspools + verbs: + - get + - list + - watch +- apiGroups: + - metallb.io + resources: + - bfdprofiles + verbs: + - get + - list + - watch +- apiGroups: + - metallb.io + resources: + - ipaddresspools + verbs: + - get + - list + - watch +- apiGroups: + - metallb.io + resources: + - bgpadvertisements + verbs: + - get + - list + - watch +- apiGroups: + - metallb.io + resources: + - l2advertisements + verbs: + - get + - list + - watch +- apiGroups: + - metallb.io + resources: + - communities + verbs: + - get + - list + - watch +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + labels: + app: metallb + name: pod-lister + namespace: metallb-system +rules: +- apiGroups: + - "" + resources: + - pods + verbs: + - list +- apiGroups: + - "" + resources: + - secrets + verbs: + - get + - list + - watch +- apiGroups: + - metallb.io + resources: + - addresspools + verbs: + - get + - list + - watch +- apiGroups: + - metallb.io + resources: + - bfdprofiles + verbs: + - get + - list + - watch +- apiGroups: + - metallb.io + resources: + - bgppeers + verbs: + - get + - list + - watch +- apiGroups: + - metallb.io + resources: + - l2advertisements + verbs: + - get + - list + - watch +- apiGroups: + - metallb.io + resources: + - bgpadvertisements + verbs: + - get + - list + - watch +- apiGroups: + - metallb.io + resources: + - ipaddresspools + verbs: + - get + - list + - watch +- apiGroups: + - metallb.io + resources: + - communities + verbs: + - get + - list + - watch +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + app: metallb + name: metallb-system:controller +rules: +- apiGroups: + - "" + resources: + - services + - namespaces + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + - services/status + verbs: + - update +- apiGroups: + - "" + resources: + - events + verbs: + - create + - patch +- apiGroups: + - policy + resourceNames: + - controller + resources: + - podsecuritypolicies + verbs: + - use +- apiGroups: + - admissionregistration.k8s.io + resourceNames: + - metallb-webhook-configuration + resources: + - validatingwebhookconfigurations + - mutatingwebhookconfigurations + verbs: + - create + - delete + - get + - list + - patch + - update + - watch +- apiGroups: + - admissionregistration.k8s.io + resources: + - validatingwebhookconfigurations + - mutatingwebhookconfigurations + verbs: + - list + - watch +- apiGroups: + - apiextensions.k8s.io + resourceNames: + - addresspools.metallb.io + - bfdprofiles.metallb.io + - bgpadvertisements.metallb.io + - bgppeers.metallb.io + - ipaddresspools.metallb.io + - l2advertisements.metallb.io + - communities.metallb.io + resources: + - customresourcedefinitions + verbs: + - create + - delete + - get + - list + - patch + - update + - watch +- apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + verbs: + - list + - watch +--- +{% if metallb_speaker_enabled %} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + app: metallb + name: metallb-system:speaker +rules: +- apiGroups: + - "" + resources: + - services + - endpoints + - nodes + - namespaces + verbs: + - get + - list + - watch +- apiGroups: + - discovery.k8s.io + resources: + - endpointslices + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + - events + verbs: + - create + - patch +- apiGroups: + - policy + resourceNames: + - speaker + resources: + - podsecuritypolicies + verbs: + - use +{% endif %} + +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + labels: + app: metallb + name: controller + namespace: metallb-system +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: controller +subjects: +- kind: ServiceAccount + name: controller + namespace: metallb-system + +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + labels: + app: metallb + name: pod-lister + namespace: metallb-system +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: pod-lister +subjects: +- kind: ServiceAccount + name: speaker + namespace: metallb-system + +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + app: metallb + name: metallb-system:controller +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: metallb-system:controller +subjects: +- kind: ServiceAccount + name: controller + namespace: metallb-system + +{% if metallb_speaker_enabled %} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + app: metallb + name: metallb-system:speaker +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: metallb-system:speaker +subjects: +- kind: ServiceAccount + name: speaker + namespace: metallb-system +{% endif %} + +--- +apiVersion: v1 +kind: Secret +metadata: + name: webhook-server-cert + namespace: metallb-system + +--- +apiVersion: v1 +kind: Service +metadata: + name: webhook-service + namespace: metallb-system +spec: + ports: + - port: 443 + targetPort: 9443 + selector: + component: controller + +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + app: metallb + component: controller + name: controller + namespace: metallb-system +spec: + revisionHistoryLimit: 3 + selector: + matchLabels: + app: metallb + component: controller + template: + metadata: + annotations: + prometheus.io/port: '{{ metallb_port }}' + prometheus.io/scrape: 'true' + labels: + app: metallb + component: controller + spec: + priorityClassName: system-cluster-critical + containers: + - args: + - --port={{ metallb_port }} + - --log-level={{ metallb_log_level }} + env: + - name: METALLB_ML_SECRET_NAME + value: memberlist + - name: METALLB_DEPLOYMENT + value: controller + image: "{{ metallb_controller_image_repo }}:{{ metallb_version }}" + livenessProbe: + failureThreshold: 3 + httpGet: + path: /metrics + port: monitoring + initialDelaySeconds: 10 + periodSeconds: 10 + successThreshold: 1 + timeoutSeconds: 1 + name: controller + ports: + - containerPort: {{ metallb_port }} + name: monitoring + - containerPort: 9443 + name: webhook-server + protocol: TCP + readinessProbe: + failureThreshold: 3 + httpGet: + path: /metrics + port: monitoring + initialDelaySeconds: 10 + periodSeconds: 10 + successThreshold: 1 + timeoutSeconds: 1 + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - all + readOnlyRootFilesystem: true + volumeMounts: + - mountPath: /tmp/k8s-webhook-server/serving-certs + name: cert + readOnly: true +{% if metallb_config.controller is defined and metallb_config.controller.tolerations is defined %} + tolerations: + {{ metallb_config.controller.tolerations | to_nice_yaml(indent=2) | indent(width=8) }} +{%- endif %} + nodeSelector: + {{ metallb_controller_nodeselector | to_nice_yaml | indent(width=8) -}} + {% if metallb_config.controller is defined and metallb_config.controller.nodeselector is defined %} + {{ metallb_config.controller.nodeselector | to_nice_yaml | indent(width=8) -}} + {%- endif %} + securityContext: + fsGroup: 65534 + runAsNonRoot: true + runAsUser: 65534 + serviceAccountName: controller + terminationGracePeriodSeconds: 0 + volumes: + - name: cert + secret: + defaultMode: 420 + secretName: webhook-server-cert + +--- +{% if metallb_speaker_enabled %} +apiVersion: apps/v1 +kind: DaemonSet +metadata: + labels: + app: metallb + component: speaker + name: speaker + namespace: metallb-system +spec: + selector: + matchLabels: + app: metallb + component: speaker + template: + metadata: + annotations: + prometheus.io/port: '{{ metallb_port }}' + prometheus.io/scrape: 'true' + labels: + app: metallb + component: speaker + spec: + containers: + - args: + - --port={{ metallb_port }} + - --log-level={{ metallb_log_level }} + env: + - name: METALLB_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: METALLB_HOST + valueFrom: + fieldRef: + fieldPath: status.hostIP + - name: METALLB_ML_BIND_ADDR + valueFrom: + fieldRef: + fieldPath: status.podIP + - name: METALLB_ML_LABELS + value: app=metallb,component=speaker + - name: METALLB_ML_SECRET_KEY + valueFrom: + secretKeyRef: + key: secretkey + name: memberlist + image: "{{ metallb_speaker_image_repo }}:{{ metallb_version }}" + livenessProbe: + failureThreshold: 3 + httpGet: + path: /metrics + port: monitoring + initialDelaySeconds: 10 + periodSeconds: 10 + successThreshold: 1 + timeoutSeconds: 1 + name: speaker + ports: + - containerPort: {{ metallb_port }} + name: monitoring + - containerPort: {{ metallb_memberlist_port }} + name: memberlist-tcp + - containerPort: {{ metallb_memberlist_port }} + name: memberlist-udp + protocol: UDP + readinessProbe: + failureThreshold: 3 + httpGet: + path: /metrics + port: monitoring + initialDelaySeconds: 10 + periodSeconds: 10 + successThreshold: 1 + timeoutSeconds: 1 + securityContext: + allowPrivilegeEscalation: false + capabilities: + add: + - NET_RAW + drop: + - ALL + readOnlyRootFilesystem: true + hostNetwork: true + nodeSelector: + {{ metallb_speaker_nodeselector | to_nice_yaml | indent(width=8) -}} + {% if metallb_config.speaker is defined and metallb_config.speaker.nodeselector is defined %} + {{ metallb_config.speaker.nodeselector | to_nice_yaml | indent(width=8) -}} + {%- endif %} + + serviceAccountName: speaker + terminationGracePeriodSeconds: 2 + tolerations: + {{ metallb_speaker_tolerations | to_nice_yaml(indent=2) | indent(width=8) -}} + {% if metallb_config.speaker is defined and metallb_config.speaker.tolerations is defined %} + {{ metallb_config.speaker.tolerations | to_nice_yaml(indent=2) | indent(width=8) -}} + {% endif %} +{% endif %} + +--- +apiVersion: admissionregistration.k8s.io/v1 +kind: ValidatingWebhookConfiguration +metadata: + creationTimestamp: null + name: metallb-webhook-configuration +webhooks: +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: webhook-service + namespace: metallb-system + path: /validate-metallb-io-v1beta2-bgppeer + failurePolicy: Fail + name: bgppeersvalidationwebhook.metallb.io + rules: + - apiGroups: + - metallb.io + apiVersions: + - v1beta2 + operations: + - CREATE + - UPDATE + resources: + - bgppeers + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: webhook-service + namespace: metallb-system + path: /validate-metallb-io-v1beta1-addresspool + failurePolicy: Fail + name: addresspoolvalidationwebhook.metallb.io + rules: + - apiGroups: + - metallb.io + apiVersions: + - v1beta1 + operations: + - CREATE + - UPDATE + resources: + - addresspools + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: webhook-service + namespace: metallb-system + path: /validate-metallb-io-v1beta1-bfdprofile + failurePolicy: Fail + name: bfdprofilevalidationwebhook.metallb.io + rules: + - apiGroups: + - metallb.io + apiVersions: + - v1beta1 + operations: + - CREATE + - DELETE + resources: + - bfdprofiles + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: webhook-service + namespace: metallb-system + path: /validate-metallb-io-v1beta1-bgpadvertisement + failurePolicy: Fail + name: bgpadvertisementvalidationwebhook.metallb.io + rules: + - apiGroups: + - metallb.io + apiVersions: + - v1beta1 + operations: + - CREATE + - UPDATE + resources: + - bgpadvertisements + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: webhook-service + namespace: metallb-system + path: /validate-metallb-io-v1beta1-community + failurePolicy: Fail + name: communityvalidationwebhook.metallb.io + rules: + - apiGroups: + - metallb.io + apiVersions: + - v1beta1 + operations: + - CREATE + - UPDATE + resources: + - communities + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: webhook-service + namespace: metallb-system + path: /validate-metallb-io-v1beta1-ipaddresspool + failurePolicy: Fail + name: ipaddresspoolvalidationwebhook.metallb.io + rules: + - apiGroups: + - metallb.io + apiVersions: + - v1beta1 + operations: + - CREATE + - UPDATE + resources: + - ipaddresspools + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: webhook-service + namespace: metallb-system + path: /validate-metallb-io-v1beta1-l2advertisement + failurePolicy: Fail + name: l2advertisementvalidationwebhook.metallb.io + rules: + - apiGroups: + - metallb.io + apiVersions: + - v1beta1 + operations: + - CREATE + - UPDATE + resources: + - l2advertisements + sideEffects: None diff --git a/kubespray/roles/kubernetes-apps/metallb/templates/pools.yaml.j2 b/kubespray/roles/kubernetes-apps/metallb/templates/pools.yaml.j2 new file mode 100644 index 0000000..f22a4e3 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metallb/templates/pools.yaml.j2 @@ -0,0 +1,22 @@ +#jinja2: trim_blocks: True, lstrip_blocks: True +# yamllint disable-file +--- + +# Create all pools +{% for pool_name, pool in metallb_config.address_pools.items() %} + +--- +apiVersion: metallb.io/v1beta1 +kind: IPAddressPool +metadata: + namespace: metallb-system + name: "{{ pool_name }}" +spec: + addresses: +{% for ip_range in pool.ip_range %} + - "{{ ip_range }}" +{% endfor %} + autoAssign: {{ pool.auto_assign | default(true) }} + avoidBuggyIPs: {{ pool.avoid_buggy_ips | default(false) }} + +{% endfor %} diff --git a/kubespray/roles/kubernetes-apps/metrics_server/defaults/main.yml b/kubespray/roles/kubernetes-apps/metrics_server/defaults/main.yml new file mode 100644 index 0000000..934e67b --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metrics_server/defaults/main.yml @@ -0,0 +1,14 @@ +--- +metrics_server_container_port: 10250 +metrics_server_kubelet_insecure_tls: true +metrics_server_kubelet_preferred_address_types: "InternalIP,ExternalIP,Hostname" +metrics_server_metric_resolution: 15s +metrics_server_limits_cpu: 100m +metrics_server_limits_memory: 200Mi +metrics_server_requests_cpu: 100m +metrics_server_requests_memory: 200Mi +metrics_server_host_network: false +metrics_server_replicas: 1 +# metrics_server_extra_tolerations: [] +# metrics_server_extra_affinity: {} +# metrics_server_nodeselector: {} diff --git a/kubespray/roles/kubernetes-apps/metrics_server/tasks/main.yml b/kubespray/roles/kubernetes-apps/metrics_server/tasks/main.yml new file mode 100644 index 0000000..1fe617d --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metrics_server/tasks/main.yml @@ -0,0 +1,57 @@ +--- +# If all masters have node role, there are no tainted master and toleration should not be specified. +- name: Check all masters are node or not + set_fact: + masters_are_not_tainted: "{{ groups['kube_node'] | intersect(groups['kube_control_plane']) == groups['kube_control_plane'] }}" + +- name: Metrics Server | Delete addon dir + file: + path: "{{ kube_config_dir }}/addons/metrics_server" + state: absent + when: + - inventory_hostname == groups['kube_control_plane'][0] + tags: + - upgrade + +- name: Metrics Server | Create addon dir + file: + path: "{{ kube_config_dir }}/addons/metrics_server" + state: directory + owner: root + group: root + mode: 0755 + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Metrics Server | Templates list + set_fact: + metrics_server_templates: + - { name: auth-delegator, file: auth-delegator.yaml, type: clusterrolebinding } + - { name: auth-reader, file: auth-reader.yaml, type: rolebinding } + - { name: metrics-server-sa, file: metrics-server-sa.yaml, type: sa } + - { name: metrics-server-deployment, file: metrics-server-deployment.yaml, type: deploy } + - { name: metrics-server-service, file: metrics-server-service.yaml, type: service } + - { name: metrics-apiservice, file: metrics-apiservice.yaml, type: service } + - { name: resource-reader-clusterrolebinding, file: resource-reader-clusterrolebinding.yaml, type: clusterrolebinding } + - { name: resource-reader, file: resource-reader.yaml, type: clusterrole } + +- name: Metrics Server | Create manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/metrics_server/{{ item.file }}" + mode: 0644 + with_items: "{{ metrics_server_templates }}" + register: metrics_server_manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Metrics Server | Apply manifests + kube: + name: "{{ item.item.name }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/metrics_server/{{ item.item.file }}" + state: "latest" + with_items: "{{ metrics_server_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/metrics_server/templates/auth-delegator.yaml.j2 b/kubespray/roles/kubernetes-apps/metrics_server/templates/auth-delegator.yaml.j2 new file mode 100644 index 0000000..92f8204 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metrics_server/templates/auth-delegator.yaml.j2 @@ -0,0 +1,14 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: metrics-server:system:auth-delegator + labels: + addonmanager.kubernetes.io/mode: Reconcile +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: system:auth-delegator +subjects: +- kind: ServiceAccount + name: metrics-server + namespace: kube-system diff --git a/kubespray/roles/kubernetes-apps/metrics_server/templates/auth-reader.yaml.j2 b/kubespray/roles/kubernetes-apps/metrics_server/templates/auth-reader.yaml.j2 new file mode 100644 index 0000000..e02b8ea --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metrics_server/templates/auth-reader.yaml.j2 @@ -0,0 +1,15 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: metrics-server-auth-reader + namespace: kube-system + labels: + addonmanager.kubernetes.io/mode: Reconcile +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: extension-apiserver-authentication-reader +subjects: +- kind: ServiceAccount + name: metrics-server + namespace: kube-system diff --git a/kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-apiservice.yaml.j2 b/kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-apiservice.yaml.j2 new file mode 100644 index 0000000..9341687 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-apiservice.yaml.j2 @@ -0,0 +1,15 @@ +apiVersion: apiregistration.k8s.io/v1 +kind: APIService +metadata: + name: v1beta1.metrics.k8s.io + labels: + addonmanager.kubernetes.io/mode: Reconcile +spec: + service: + name: metrics-server + namespace: kube-system + group: metrics.k8s.io + version: v1beta1 + insecureSkipTLSVerify: {{ metrics_server_kubelet_insecure_tls }} + groupPriorityMinimum: 100 + versionPriority: 100 diff --git a/kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-server-deployment.yaml.j2 b/kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-server-deployment.yaml.j2 new file mode 100644 index 0000000..9bee26b --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-server-deployment.yaml.j2 @@ -0,0 +1,121 @@ +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + name: metrics-server + namespace: kube-system + labels: + app.kubernetes.io/name: metrics-server + addonmanager.kubernetes.io/mode: Reconcile + version: {{ metrics_server_version }} +spec: + replicas: {{ metrics_server_replicas }} + selector: + matchLabels: + app.kubernetes.io/name: metrics-server + version: {{ metrics_server_version }} + strategy: + rollingUpdate: + maxUnavailable: 0 + template: + metadata: + name: metrics-server + labels: + app.kubernetes.io/name: metrics-server + version: {{ metrics_server_version }} + spec: + priorityClassName: system-cluster-critical + serviceAccountName: metrics-server + hostNetwork: {{ metrics_server_host_network | default(false) }} + containers: + - name: metrics-server + image: {{ metrics_server_image_repo }}:{{ metrics_server_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - --logtostderr + - --cert-dir=/tmp + - --secure-port={{ metrics_server_container_port }} +{% if metrics_server_kubelet_preferred_address_types %} + - --kubelet-preferred-address-types={{ metrics_server_kubelet_preferred_address_types }} +{% endif %} + - --kubelet-use-node-status-port +{% if metrics_server_kubelet_insecure_tls %} + - --kubelet-insecure-tls=true +{% endif %} + - --metric-resolution={{ metrics_server_metric_resolution }} + ports: + - containerPort: {{ metrics_server_container_port }} + name: https + protocol: TCP + volumeMounts: + - name: tmp + mountPath: /tmp + livenessProbe: + httpGet: + path: /livez + port: https + scheme: HTTPS + periodSeconds: 10 + failureThreshold: 3 + initialDelaySeconds: 40 + readinessProbe: + httpGet: + path: /readyz + port: https + scheme: HTTPS + periodSeconds: 10 + failureThreshold: 3 + initialDelaySeconds: 40 + securityContext: + readOnlyRootFilesystem: true + runAsNonRoot: true + runAsUser: 1000 + allowPrivilegeEscalation: false + seccompProfile: + type: RuntimeDefault + capabilities: + drop: + - ALL + resources: + limits: + cpu: {{ metrics_server_limits_cpu }} + memory: {{ metrics_server_limits_memory }} + requests: + cpu: {{ metrics_server_requests_cpu }} + memory: {{ metrics_server_requests_memory }} + volumes: + - name: tmp + emptyDir: {} +{% if not masters_are_not_tainted or metrics_server_extra_tolerations is defined %} + tolerations: +{% if not masters_are_not_tainted %} + - key: node-role.kubernetes.io/master + effect: NoSchedule + - key: node-role.kubernetes.io/control-plane + effect: NoSchedule +{% endif %} +{% if metrics_server_extra_tolerations is defined %} + {{ metrics_server_extra_tolerations | list | to_nice_yaml(indent=2) | indent(8) }} +{% endif %} +{% endif %} + affinity: + podAntiAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - weight: 100 + podAffinityTerm: + labelSelector: + matchExpressions: + - key: app.kubernetes.io/name + operator: In + values: + - metrics-server + topologyKey: kubernetes.io/hostname + namespaces: + - kube-system +{% if metrics_server_extra_affinity is defined %} + {{ metrics_server_extra_affinity | to_nice_yaml | indent(width=8) }} +{% endif %} +{% if metrics_server_nodeselector is defined %} + nodeSelector: + {{ metrics_server_nodeselector | to_nice_yaml | indent(width=8) }} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-server-sa.yaml.j2 b/kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-server-sa.yaml.j2 new file mode 100644 index 0000000..94444ca --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-server-sa.yaml.j2 @@ -0,0 +1,8 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: metrics-server + namespace: kube-system + labels: + addonmanager.kubernetes.io/mode: Reconcile diff --git a/kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-server-service.yaml.j2 b/kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-server-service.yaml.j2 new file mode 100644 index 0000000..f1c3691 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metrics_server/templates/metrics-server-service.yaml.j2 @@ -0,0 +1,17 @@ +apiVersion: v1 +kind: Service +metadata: + name: metrics-server + namespace: kube-system + labels: + addonmanager.kubernetes.io/mode: Reconcile + app.kubernetes.io/name: "metrics-server" +spec: + type: ClusterIP + selector: + app.kubernetes.io/name: metrics-server + ports: + - name: https + port: 443 + protocol: TCP + targetPort: https diff --git a/kubespray/roles/kubernetes-apps/metrics_server/templates/resource-reader-clusterrolebinding.yaml.j2 b/kubespray/roles/kubernetes-apps/metrics_server/templates/resource-reader-clusterrolebinding.yaml.j2 new file mode 100644 index 0000000..038cfd8 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metrics_server/templates/resource-reader-clusterrolebinding.yaml.j2 @@ -0,0 +1,15 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: system:metrics-server + labels: + addonmanager.kubernetes.io/mode: Reconcile +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: system:metrics-server +subjects: +- kind: ServiceAccount + name: metrics-server + namespace: kube-system diff --git a/kubespray/roles/kubernetes-apps/metrics_server/templates/resource-reader.yaml.j2 b/kubespray/roles/kubernetes-apps/metrics_server/templates/resource-reader.yaml.j2 new file mode 100644 index 0000000..9c4a3b7 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/metrics_server/templates/resource-reader.yaml.j2 @@ -0,0 +1,22 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: system:metrics-server + labels: + addonmanager.kubernetes.io/mode: Reconcile +rules: +- apiGroups: + - "" + resources: + - nodes/metrics + verbs: + - get +- apiGroups: + - "" + resources: + - pods + - nodes + verbs: + - get + - list + - watch diff --git a/kubespray/roles/kubernetes-apps/network_plugin/calico/tasks/main.yml b/kubespray/roles/kubernetes-apps/network_plugin/calico/tasks/main.yml new file mode 100644 index 0000000..b8b4338 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/network_plugin/calico/tasks/main.yml @@ -0,0 +1,2 @@ +--- +# TODO: Handle Calico etcd -> kdd migration diff --git a/kubespray/roles/kubernetes-apps/network_plugin/flannel/tasks/main.yml b/kubespray/roles/kubernetes-apps/network_plugin/flannel/tasks/main.yml new file mode 100644 index 0000000..ff56d24 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/network_plugin/flannel/tasks/main.yml @@ -0,0 +1,17 @@ +--- +- name: Flannel | Start Resources + kube: + name: "{{ item.item.name }}" + namespace: "kube-system" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: "{{ flannel_node_manifests.results }}" + when: inventory_hostname == groups['kube_control_plane'][0] and not item is skipped + +- name: Flannel | Wait for flannel subnet.env file presence + wait_for: + path: /run/flannel/subnet.env + delay: 5 + timeout: 600 diff --git a/kubespray/roles/kubernetes-apps/network_plugin/kube-ovn/tasks/main.yml b/kubespray/roles/kubernetes-apps/network_plugin/kube-ovn/tasks/main.yml new file mode 100644 index 0000000..9f42501 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/network_plugin/kube-ovn/tasks/main.yml @@ -0,0 +1,9 @@ +--- +- name: Kube-OVN | Start Resources + kube: + name: "{{ item.item.name }}" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: "{{ kube_ovn_node_manifests.results }}" + when: inventory_hostname == groups['kube_control_plane'][0] and not item is skipped diff --git a/kubespray/roles/kubernetes-apps/network_plugin/kube-router/OWNERS b/kubespray/roles/kubernetes-apps/network_plugin/kube-router/OWNERS new file mode 100644 index 0000000..c40af3c --- /dev/null +++ b/kubespray/roles/kubernetes-apps/network_plugin/kube-router/OWNERS @@ -0,0 +1,6 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +approvers: + - bozzo +reviewers: + - bozzo diff --git a/kubespray/roles/kubernetes-apps/network_plugin/kube-router/tasks/main.yml b/kubespray/roles/kubernetes-apps/network_plugin/kube-router/tasks/main.yml new file mode 100644 index 0000000..1d756a0 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/network_plugin/kube-router/tasks/main.yml @@ -0,0 +1,23 @@ +--- + +- name: Kube-router | Start Resources + kube: + name: "kube-router" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/kube-router.yml" + resource: "ds" + namespace: "kube-system" + state: "latest" + delegate_to: "{{ groups['kube_control_plane'] | first }}" + run_once: true + +- name: Kube-router | Wait for kube-router pods to be ready + command: "{{ kubectl }} -n kube-system get pods -l k8s-app=kube-router -o jsonpath='{.items[?(@.status.containerStatuses[0].ready==false)].metadata.name}'" # noqa ignore-errors + register: pods_not_ready + until: pods_not_ready.stdout.find("kube-router")==-1 + retries: 30 + delay: 10 + ignore_errors: true + delegate_to: "{{ groups['kube_control_plane'] | first }}" + run_once: true + changed_when: false diff --git a/kubespray/roles/kubernetes-apps/network_plugin/meta/main.yml b/kubespray/roles/kubernetes-apps/network_plugin/meta/main.yml new file mode 100644 index 0000000..1128312 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/network_plugin/meta/main.yml @@ -0,0 +1,31 @@ +--- +dependencies: + - role: kubernetes-apps/network_plugin/calico + when: kube_network_plugin == 'calico' + tags: + - calico + + - role: kubernetes-apps/network_plugin/flannel + when: kube_network_plugin == 'flannel' + tags: + - flannel + + - role: kubernetes-apps/network_plugin/kube-ovn + when: kube_network_plugin == 'kube-ovn' + tags: + - kube-ovn + + - role: kubernetes-apps/network_plugin/weave + when: kube_network_plugin == 'weave' + tags: + - weave + + - role: kubernetes-apps/network_plugin/kube-router + when: kube_network_plugin == 'kube-router' + tags: + - kube-router + + - role: kubernetes-apps/network_plugin/multus + when: kube_network_plugin_multus + tags: + - multus diff --git a/kubespray/roles/kubernetes-apps/network_plugin/multus/tasks/main.yml b/kubespray/roles/kubernetes-apps/network_plugin/multus/tasks/main.yml new file mode 100644 index 0000000..fdbede5 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/network_plugin/multus/tasks/main.yml @@ -0,0 +1,18 @@ +--- +- name: Multus | Start resources + kube: + name: "{{ item.item.name }}" + namespace: "kube-system" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + delegate_to: "{{ groups['kube_control_plane'][0] }}" + run_once: true + with_items: "{{ multus_manifest_1.results + (multus_nodes_list | map('extract', hostvars, 'multus_manifest_2') | list | json_query('[].results')) }}" + loop_control: + label: "{{ item.item.name if item != None else 'skipped' }}" + vars: + multus_nodes_list: "{{ groups['k8s_cluster'] if ansible_play_batch | length == ansible_play_hosts_all | length else ansible_play_batch }}" + when: + - not item is skipped diff --git a/kubespray/roles/kubernetes-apps/network_plugin/weave/tasks/main.yml b/kubespray/roles/kubernetes-apps/network_plugin/weave/tasks/main.yml new file mode 100644 index 0000000..bc0f932 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/network_plugin/weave/tasks/main.yml @@ -0,0 +1,21 @@ +--- + +- name: Weave | Start Resources + kube: + name: "weave-net" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/weave-net.yml" + resource: "ds" + namespace: "kube-system" + state: "latest" + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Weave | Wait for Weave to become available + uri: + url: http://127.0.0.1:6784/status + return_content: yes + register: weave_status + retries: 180 + delay: 5 + until: "weave_status.status == 200 and 'Status: ready' in weave_status.content" + when: inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/OWNERS b/kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/OWNERS new file mode 100644 index 0000000..6e44ceb --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/OWNERS @@ -0,0 +1,5 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +approvers: + - alijahnas +reviewers: diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/defaults/main.yml b/kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/defaults/main.yml new file mode 100644 index 0000000..896d2d3 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/defaults/main.yml @@ -0,0 +1,8 @@ +--- +# To restrict which AZ the volume should be provisioned in +# set this value to true and set the list of relevant AZs +# For it to work, the flag aws_ebs_csi_enable_volume_scheduling +# in AWS EBS Driver must be true +restrict_az_provisioning: false +aws_ebs_availability_zones: + - eu-west-3c diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/tasks/main.yml b/kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/tasks/main.yml new file mode 100644 index 0000000..b49acdf --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/tasks/main.yml @@ -0,0 +1,20 @@ +--- +- name: Kubernetes Persistent Volumes | Copy AWS EBS CSI Storage Class template + template: + src: "aws-ebs-csi-storage-class.yml.j2" + dest: "{{ kube_config_dir }}/aws-ebs-csi-storage-class.yml" + mode: 0644 + register: manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Persistent Volumes | Add AWS EBS CSI Storage Class + kube: + name: aws-ebs-csi + kubectl: "{{ bin_dir }}/kubectl" + resource: StorageClass + filename: "{{ kube_config_dir }}/aws-ebs-csi-storage-class.yml" + state: "latest" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - manifests.changed diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/templates/aws-ebs-csi-storage-class.yml.j2 b/kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/templates/aws-ebs-csi-storage-class.yml.j2 new file mode 100644 index 0000000..1632646 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/aws-ebs-csi/templates/aws-ebs-csi-storage-class.yml.j2 @@ -0,0 +1,18 @@ +kind: StorageClass +apiVersion: storage.k8s.io/v1 +metadata: + name: ebs-sc +provisioner: ebs.csi.aws.com +volumeBindingMode: WaitForFirstConsumer +parameters: + csi.storage.k8s.io/fstype: xfs + type: gp2 +{% if restrict_az_provisioning %} +allowedTopologies: +- matchLabelExpressions: + - key: topology.ebs.csi.aws.com/zone + values: +{% for value in aws_ebs_availability_zones %} + - {{ value }} +{% endfor %} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/azuredisk-csi/defaults/main.yml b/kubespray/roles/kubernetes-apps/persistent_volumes/azuredisk-csi/defaults/main.yml new file mode 100644 index 0000000..fc92e17 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/azuredisk-csi/defaults/main.yml @@ -0,0 +1,3 @@ +--- +## Available values: Standard_LRS, Premium_LRS, StandardSSD_LRS, UltraSSD_LRS +storage_account_type: StandardSSD_LRS diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/azuredisk-csi/tasks/main.yml b/kubespray/roles/kubernetes-apps/persistent_volumes/azuredisk-csi/tasks/main.yml new file mode 100644 index 0000000..9abffbe --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/azuredisk-csi/tasks/main.yml @@ -0,0 +1,20 @@ +--- +- name: Kubernetes Persistent Volumes | Copy Azure CSI Storage Class template + template: + src: "azure-csi-storage-class.yml.j2" + dest: "{{ kube_config_dir }}/azure-csi-storage-class.yml" + mode: 0644 + register: manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Persistent Volumes | Add Azure CSI Storage Class + kube: + name: azure-csi + kubectl: "{{ bin_dir }}/kubectl" + resource: StorageClass + filename: "{{ kube_config_dir }}/azure-csi-storage-class.yml" + state: "latest" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - manifests.changed diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/azuredisk-csi/templates/azure-csi-storage-class.yml.j2 b/kubespray/roles/kubernetes-apps/persistent_volumes/azuredisk-csi/templates/azure-csi-storage-class.yml.j2 new file mode 100644 index 0000000..be5cb38 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/azuredisk-csi/templates/azure-csi-storage-class.yml.j2 @@ -0,0 +1,14 @@ +--- +apiVersion: storage.k8s.io/v1 +kind: StorageClass +metadata: + name: disk.csi.azure.com +provisioner: disk.csi.azure.com +parameters: + skuname: {{ storage_account_type }} +{% if azure_csi_tags is defined %} + tags: {{ azure_csi_tags }} +{% endif %} +reclaimPolicy: Delete +volumeBindingMode: Immediate +allowVolumeExpansion: true diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/cinder-csi/defaults/main.yml b/kubespray/roles/kubernetes-apps/persistent_volumes/cinder-csi/defaults/main.yml new file mode 100644 index 0000000..5e35dd5 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/cinder-csi/defaults/main.yml @@ -0,0 +1,7 @@ +--- +storage_classes: + - name: cinder-csi + is_default: false + parameters: + availability: nova + allowVolumeExpansion: false diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/cinder-csi/tasks/main.yml b/kubespray/roles/kubernetes-apps/persistent_volumes/cinder-csi/tasks/main.yml new file mode 100644 index 0000000..52de1c5 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/cinder-csi/tasks/main.yml @@ -0,0 +1,20 @@ +--- +- name: Kubernetes Persistent Volumes | Copy Cinder CSI Storage Class template + template: + src: "cinder-csi-storage-class.yml.j2" + dest: "{{ kube_config_dir }}/cinder-csi-storage-class.yml" + mode: 0644 + register: manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Persistent Volumes | Add Cinder CSI Storage Class + kube: + name: cinder-csi + kubectl: "{{ bin_dir }}/kubectl" + resource: StorageClass + filename: "{{ kube_config_dir }}/cinder-csi-storage-class.yml" + state: "latest" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - manifests.changed diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/cinder-csi/templates/cinder-csi-storage-class.yml.j2 b/kubespray/roles/kubernetes-apps/persistent_volumes/cinder-csi/templates/cinder-csi-storage-class.yml.j2 new file mode 100644 index 0000000..be8ba13 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/cinder-csi/templates/cinder-csi-storage-class.yml.j2 @@ -0,0 +1,25 @@ +{% for class in storage_classes %} +--- +kind: StorageClass +apiVersion: storage.k8s.io/v1 +metadata: + name: "{{ class.name }}" + annotations: + storageclass.kubernetes.io/is-default-class: "{{ class.is_default | default(false) | ternary("true","false") }}" +provisioner: cinder.csi.openstack.org +volumeBindingMode: WaitForFirstConsumer +parameters: +{% for key, value in (class.parameters | default({})).items() %} + "{{ key }}": "{{ value }}" +{% endfor %} +{% if cinder_topology is defined and cinder_topology is sameas true %} +allowedTopologies: +- matchLabelExpressions: + - key: topology.cinder.csi.openstack.org/zone + values: +{% for zone in cinder_topology_zones %} + - "{{ zone }}" +{% endfor %} +{% endif %} +allowVolumeExpansion: {{ expand_persistent_volumes }} +{% endfor %} diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/gcp-pd-csi/defaults/main.yml b/kubespray/roles/kubernetes-apps/persistent_volumes/gcp-pd-csi/defaults/main.yml new file mode 100644 index 0000000..d58706f --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/gcp-pd-csi/defaults/main.yml @@ -0,0 +1,8 @@ +--- +# Choose between pd-standard and pd-ssd +gcp_pd_csi_volume_type: pd-standard +gcp_pd_regional_replication_enabled: false +gcp_pd_restrict_zone_replication: false +gcp_pd_restricted_zones: + - europe-west1-b + - europe-west1-c diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/gcp-pd-csi/tasks/main.yml b/kubespray/roles/kubernetes-apps/persistent_volumes/gcp-pd-csi/tasks/main.yml new file mode 100644 index 0000000..29997e7 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/gcp-pd-csi/tasks/main.yml @@ -0,0 +1,20 @@ +--- +- name: Kubernetes Persistent Volumes | Copy GCP PD CSI Storage Class template + template: + src: "gcp-pd-csi-storage-class.yml.j2" + dest: "{{ kube_config_dir }}/gcp-pd-csi-storage-class.yml" + mode: 0644 + register: manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Persistent Volumes | Add GCP PD CSI Storage Class + kube: + name: gcp-pd-csi + kubectl: "{{ bin_dir }}/kubectl" + resource: StorageClass + filename: "{{ kube_config_dir }}/gcp-pd-csi-storage-class.yml" + state: "latest" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - manifests.changed diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/gcp-pd-csi/templates/gcp-pd-csi-storage-class.yml.j2 b/kubespray/roles/kubernetes-apps/persistent_volumes/gcp-pd-csi/templates/gcp-pd-csi-storage-class.yml.j2 new file mode 100644 index 0000000..475eb4f --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/gcp-pd-csi/templates/gcp-pd-csi-storage-class.yml.j2 @@ -0,0 +1,20 @@ +apiVersion: storage.k8s.io/v1 +kind: StorageClass +metadata: + name: csi-gce-pd +provisioner: pd.csi.storage.gke.io +parameters: + type: {{ gcp_pd_csi_volume_type }} +{% if gcp_pd_regional_replication_enabled %} + replication-type: regional-pd +{% endif %} +volumeBindingMode: WaitForFirstConsumer +{% if gcp_pd_restrict_zone_replication %} +allowedTopologies: +- matchLabelExpressions: + - key: topology.gke.io/zone + values: +{% for value in gcp_pd_restricted_zones %} + - {{ value }} +{% endfor %} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/meta/main.yml b/kubespray/roles/kubernetes-apps/persistent_volumes/meta/main.yml new file mode 100644 index 0000000..e3066bb --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/meta/main.yml @@ -0,0 +1,43 @@ +--- +dependencies: + - role: kubernetes-apps/persistent_volumes/openstack + when: + - cloud_provider is defined + - cloud_provider in [ 'openstack' ] + tags: + - persistent_volumes_openstack + + - role: kubernetes-apps/persistent_volumes/cinder-csi + when: + - cinder_csi_enabled + tags: + - persistent_volumes_cinder_csi + - cinder-csi-driver + + - role: kubernetes-apps/persistent_volumes/aws-ebs-csi + when: + - aws_ebs_csi_enabled + tags: + - persistent_volumes_aws_ebs_csi + - aws-ebs-csi-driver + + - role: kubernetes-apps/persistent_volumes/azuredisk-csi + when: + - azure_csi_enabled + tags: + - persistent_volumes_azure_csi + - azure-csi-driver + + - role: kubernetes-apps/persistent_volumes/gcp-pd-csi + when: + - gcp_pd_csi_enabled + tags: + - persistent_volumes_gcp_pd_csi + - gcp-pd-csi-driver + + - role: kubernetes-apps/persistent_volumes/upcloud-csi + when: + - upcloud_csi_enabled + tags: + - persistent_volumes_upcloud_csi + - upcloud-csi-driver diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/openstack/defaults/main.yml b/kubespray/roles/kubernetes-apps/persistent_volumes/openstack/defaults/main.yml new file mode 100644 index 0000000..05a3d94 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/openstack/defaults/main.yml @@ -0,0 +1,7 @@ +--- +persistent_volumes_enabled: false +storage_classes: + - name: standard + is_default: true + parameters: + availability: nova diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/openstack/tasks/main.yml b/kubespray/roles/kubernetes-apps/persistent_volumes/openstack/tasks/main.yml new file mode 100644 index 0000000..3387e7f --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/openstack/tasks/main.yml @@ -0,0 +1,20 @@ +--- +- name: Kubernetes Persistent Volumes | Lay down OpenStack Cinder Storage Class template + template: + src: "openstack-storage-class.yml.j2" + dest: "{{ kube_config_dir }}/openstack-storage-class.yml" + mode: 0644 + register: manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Persistent Volumes | Add OpenStack Cinder Storage Class + kube: + name: storage-class + kubectl: "{{ bin_dir }}/kubectl" + resource: StorageClass + filename: "{{ kube_config_dir }}/openstack-storage-class.yml" + state: "latest" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - manifests.changed diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/openstack/templates/openstack-storage-class.yml.j2 b/kubespray/roles/kubernetes-apps/persistent_volumes/openstack/templates/openstack-storage-class.yml.j2 new file mode 100644 index 0000000..0551e15 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/openstack/templates/openstack-storage-class.yml.j2 @@ -0,0 +1,15 @@ +{% for class in storage_classes %} +--- +kind: StorageClass +apiVersion: storage.k8s.io/v1 +metadata: + name: "{{ class.name }}" + annotations: + storageclass.kubernetes.io/is-default-class: "{{ class.is_default | default(false) | ternary("true","false") }}" +provisioner: kubernetes.io/cinder +parameters: +{% for key, value in (class.parameters | default({})).items() %} + "{{ key }}": "{{ value }}" +{% endfor %} +allowVolumeExpansion: {{ expand_persistent_volumes }} +{% endfor %} diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/upcloud-csi/defaults/main.yml b/kubespray/roles/kubernetes-apps/persistent_volumes/upcloud-csi/defaults/main.yml new file mode 100644 index 0000000..5986e8c --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/upcloud-csi/defaults/main.yml @@ -0,0 +1,12 @@ +--- +storage_classes: + - name: standard + is_default: true + expand_persistent_volumes: true + parameters: + tier: maxiops + - name: hdd + is_default: false + expand_persistent_volumes: true + parameters: + tier: hdd diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/upcloud-csi/tasks/main.yml b/kubespray/roles/kubernetes-apps/persistent_volumes/upcloud-csi/tasks/main.yml new file mode 100644 index 0000000..26104a0 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/upcloud-csi/tasks/main.yml @@ -0,0 +1,20 @@ +--- +- name: Kubernetes Persistent Volumes | Copy UpCloud CSI Storage Class template + template: + src: "upcloud-csi-storage-class.yml.j2" + dest: "{{ kube_config_dir }}/upcloud-csi-storage-class.yml" + mode: 0644 + register: manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Persistent Volumes | Add UpCloud CSI Storage Class + kube: + name: upcloud-csi + kubectl: "{{ bin_dir }}/kubectl" + resource: StorageClass + filename: "{{ kube_config_dir }}/upcloud-csi-storage-class.yml" + state: "latest" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - manifests.changed diff --git a/kubespray/roles/kubernetes-apps/persistent_volumes/upcloud-csi/templates/upcloud-csi-storage-class.yml.j2 b/kubespray/roles/kubernetes-apps/persistent_volumes/upcloud-csi/templates/upcloud-csi-storage-class.yml.j2 new file mode 100644 index 0000000..a40df9b --- /dev/null +++ b/kubespray/roles/kubernetes-apps/persistent_volumes/upcloud-csi/templates/upcloud-csi-storage-class.yml.j2 @@ -0,0 +1,16 @@ +{% for class in storage_classes %} +--- +kind: StorageClass +apiVersion: storage.k8s.io/v1 +metadata: + name: "{{ class.name }}" + annotations: + storageclass.kubernetes.io/is-default-class: "{{ class.is_default | default(false) | ternary("true","false") }}" +provisioner: storage.csi.upcloud.com +reclaimPolicy: Delete +parameters: +{% for key, value in (class.parameters | default({})).items() %} + "{{ key }}": "{{ value }}" +{% endfor %} +allowVolumeExpansion: {{ class.expand_persistent_volumes | default(true) | ternary("true","false") }} +{% endfor %} diff --git a/kubespray/roles/kubernetes-apps/policy_controller/calico/defaults/main.yml b/kubespray/roles/kubernetes-apps/policy_controller/calico/defaults/main.yml new file mode 100644 index 0000000..d3a780c --- /dev/null +++ b/kubespray/roles/kubernetes-apps/policy_controller/calico/defaults/main.yml @@ -0,0 +1,10 @@ +--- +# Limits for calico apps +calico_policy_controller_cpu_limit: 1000m +calico_policy_controller_memory_limit: 256M +calico_policy_controller_cpu_requests: 30m +calico_policy_controller_memory_requests: 64M +calico_policy_controller_deployment_nodeselector: "kubernetes.io/os: linux" + +# SSL +calico_cert_dir: "/etc/calico/certs" diff --git a/kubespray/roles/kubernetes-apps/policy_controller/calico/tasks/main.yml b/kubespray/roles/kubernetes-apps/policy_controller/calico/tasks/main.yml new file mode 100644 index 0000000..ba2eebb --- /dev/null +++ b/kubespray/roles/kubernetes-apps/policy_controller/calico/tasks/main.yml @@ -0,0 +1,34 @@ +--- +- name: Create calico-kube-controllers manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: calico-kube-controllers, file: calico-kube-controllers.yml, type: deployment} + - {name: calico-kube-controllers, file: calico-kube-sa.yml, type: sa} + - {name: calico-kube-controllers, file: calico-kube-cr.yml, type: clusterrole} + - {name: calico-kube-controllers, file: calico-kube-crb.yml, type: clusterrolebinding} + register: calico_kube_manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + - rbac_enabled or item.type not in rbac_resources + +- name: Start of Calico kube controllers + kube: + name: "{{ item.item.name }}" + namespace: "kube-system" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ calico_kube_manifests.results }}" + register: calico_kube_controller_start + until: calico_kube_controller_start is succeeded + retries: 4 + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item.file }}" diff --git a/kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-controllers.yml.j2 b/kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-controllers.yml.j2 new file mode 100644 index 0000000..f89e4d6 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-controllers.yml.j2 @@ -0,0 +1,87 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: calico-kube-controllers + namespace: kube-system + labels: + k8s-app: calico-kube-controllers +spec: + replicas: 1 + strategy: + type: Recreate + selector: + matchLabels: + k8s-app: calico-kube-controllers + template: + metadata: + name: calico-kube-controllers + namespace: kube-system + labels: + k8s-app: calico-kube-controllers + spec: + nodeSelector: + {{ calico_policy_controller_deployment_nodeselector }} +{% if calico_datastore == "etcd" %} + hostNetwork: true +{% endif %} + serviceAccountName: calico-kube-controllers + tolerations: + - key: CriticalAddonsOnly + operator: Exists + - key: node-role.kubernetes.io/master + effect: NoSchedule + - key: node-role.kubernetes.io/control-plane + effect: NoSchedule +{% if policy_controller_extra_tolerations is defined %} + {{ policy_controller_extra_tolerations | list | to_nice_yaml(indent=2) | indent(8) }} +{% endif %} + priorityClassName: system-cluster-critical + containers: + - name: calico-kube-controllers + image: {{ calico_policy_image_repo }}:{{ calico_policy_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + resources: + limits: + cpu: {{ calico_policy_controller_cpu_limit }} + memory: {{ calico_policy_controller_memory_limit }} + requests: + cpu: {{ calico_policy_controller_cpu_requests }} + memory: {{ calico_policy_controller_memory_requests }} + livenessProbe: + exec: + command: + - /usr/bin/check-status + - -l + periodSeconds: 10 + initialDelaySeconds: 10 + failureThreshold: 6 + readinessProbe: + exec: + command: + - /usr/bin/check-status + - -r + periodSeconds: 10 + env: +{% if calico_datastore == "kdd" %} + - name: ENABLED_CONTROLLERS + value: node + - name: DATASTORE_TYPE + value: kubernetes +{% else %} + - name: ETCD_ENDPOINTS + value: "{{ etcd_access_addresses }}" + - name: ETCD_CA_CERT_FILE + value: "{{ calico_cert_dir }}/ca_cert.crt" + - name: ETCD_CERT_FILE + value: "{{ calico_cert_dir }}/cert.crt" + - name: ETCD_KEY_FILE + value: "{{ calico_cert_dir }}/key.pem" + volumeMounts: + - mountPath: {{ calico_cert_dir }} + name: etcd-certs + readOnly: true + volumes: + - hostPath: + path: {{ calico_cert_dir }} + name: etcd-certs +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-cr.yml.j2 b/kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-cr.yml.j2 new file mode 100644 index 0000000..f74b291 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-cr.yml.j2 @@ -0,0 +1,110 @@ +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: calico-kube-controllers + namespace: kube-system +rules: +{% if calico_datastore == "etcd" %} + - apiGroups: + - "" + - extensions + resources: + - pods + - namespaces + - networkpolicies + - nodes + - serviceaccounts + verbs: + - watch + - list + - get + - apiGroups: + - "" + resources: + - nodes + verbs: + - get + - apiGroups: + - networking.k8s.io + resources: + - networkpolicies + verbs: + - watch + - list +{% elif calico_datastore == "kdd" %} + # Nodes are watched to monitor for deletions. + - apiGroups: [""] + resources: + - nodes + verbs: + - watch + - list + - get + # Pods are queried to check for existence. + - apiGroups: [""] + resources: + - pods + verbs: + - watch + - list + - get + # IPAM resources are manipulated when nodes are deleted. + - apiGroups: ["crd.projectcalico.org"] + resources: + - ipreservations + verbs: + - list + # Pools are watched to maintain a mapping of blocks to IP pools. + - apiGroups: ["crd.projectcalico.org"] + resources: + - ippools + verbs: + - list + - watch + - apiGroups: ["crd.projectcalico.org"] + resources: + - blockaffinities + - ipamblocks + - ipamhandles + verbs: + - get + - list + - create + - update + - delete + - watch + # kube-controllers manages hostendpoints. + - apiGroups: ["crd.projectcalico.org"] + resources: + - hostendpoints + verbs: + - get + - list + - create + - update + - delete + # Needs access to update clusterinformations. + - apiGroups: ["crd.projectcalico.org"] + resources: + - clusterinformations + verbs: + - get + - list + - create + - update + - watch + # KubeControllersConfiguration is where it gets its config + - apiGroups: ["crd.projectcalico.org"] + resources: + - kubecontrollersconfigurations + verbs: + # read its own config + - get + # create a default if none exists + - create + # update status + - update + # watch for changes + - watch +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-crb.yml.j2 b/kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-crb.yml.j2 new file mode 100644 index 0000000..8168056 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-crb.yml.j2 @@ -0,0 +1,13 @@ +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: calico-kube-controllers +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: calico-kube-controllers +subjects: +- kind: ServiceAccount + name: calico-kube-controllers + namespace: kube-system diff --git a/kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-sa.yml.j2 b/kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-sa.yml.j2 new file mode 100644 index 0000000..269d0a1 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/policy_controller/calico/templates/calico-kube-sa.yml.j2 @@ -0,0 +1,6 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: calico-kube-controllers + namespace: kube-system diff --git a/kubespray/roles/kubernetes-apps/policy_controller/meta/main.yml b/kubespray/roles/kubernetes-apps/policy_controller/meta/main.yml new file mode 100644 index 0000000..00fa041 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/policy_controller/meta/main.yml @@ -0,0 +1,8 @@ +--- +dependencies: + - role: policy_controller/calico + when: + - kube_network_plugin in ['calico'] + - enable_network_policy + tags: + - policy-controller diff --git a/kubespray/roles/kubernetes-apps/registry/defaults/main.yml b/kubespray/roles/kubernetes-apps/registry/defaults/main.yml new file mode 100644 index 0000000..6353b7c --- /dev/null +++ b/kubespray/roles/kubernetes-apps/registry/defaults/main.yml @@ -0,0 +1,48 @@ +--- +registry_namespace: "kube-system" +registry_storage_class: "" +registry_storage_access_mode: "ReadWriteOnce" +registry_disk_size: "10Gi" +registry_port: 5000 +registry_replica_count: 1 + +# type of service: ClusterIP, LoadBalancer or NodePort +registry_service_type: "ClusterIP" +# you can specify your cluster IP address when registry_service_type is ClusterIP +registry_service_cluster_ip: "" +# you can specify your cloud provider assigned loadBalancerIP when registry_service_type is LoadBalancer +registry_service_loadbalancer_ip: "" +# annotations for managing Cloud Load Balancers +registry_service_annotations: {} +# you can specify the node port when registry_service_type is NodePort +registry_service_nodeport: "" + +# name of kubernetes secret for registry TLS certs +registry_tls_secret: "" + +registry_htpasswd: "" + +# registry configuration +# see: https://docs.docker.com/registry/configuration/#list-of-configuration-options +registry_config: + version: 0.1 + log: + fields: + service: registry + storage: + cache: + blobdescriptor: inmemory + http: + addr: :{{ registry_port }} + headers: + X-Content-Type-Options: [nosniff] + health: + storagedriver: + enabled: true + interval: 10s + threshold: 3 + +registry_ingress_annotations: {} +registry_ingress_host: "" +# name of kubernetes secret for registry ingress TLS certs +registry_ingress_tls_secret: "" diff --git a/kubespray/roles/kubernetes-apps/registry/tasks/main.yml b/kubespray/roles/kubernetes-apps/registry/tasks/main.yml new file mode 100644 index 0000000..06f1f6a --- /dev/null +++ b/kubespray/roles/kubernetes-apps/registry/tasks/main.yml @@ -0,0 +1,109 @@ +--- +- name: Registry | check registry_service_type value + fail: + msg: "registry_service_type can only be 'ClusterIP', 'LoadBalancer' or 'NodePort'" + when: registry_service_type not in ['ClusterIP', 'LoadBalancer', 'NodePort'] + +- name: Registry | Stop if registry_service_cluster_ip is defined when registry_service_type is not 'ClusterIP' + fail: + msg: "registry_service_cluster_ip support only compatible with ClusterIP." + when: + - registry_service_cluster_ip is defined and registry_service_cluster_ip | length > 0 + - registry_service_type != "ClusterIP" + +- name: Registry | Stop if registry_service_loadbalancer_ip is defined when registry_service_type is not 'LoadBalancer' + fail: + msg: "registry_service_loadbalancer_ip support only compatible with LoadBalancer." + when: + - registry_service_loadbalancer_ip is defined and registry_service_loadbalancer_ip | length > 0 + - registry_service_type != "LoadBalancer" + +- name: Registry | Stop if registry_service_nodeport is defined when registry_service_type is not 'NodePort' + fail: + msg: "registry_service_nodeport support only compatible with NodePort." + when: + - registry_service_nodeport is defined and registry_service_nodeport | length > 0 + - registry_service_type != "NodePort" + +- name: Registry | Create addon dir + file: + path: "{{ kube_config_dir }}/addons/registry" + state: directory + owner: root + group: root + mode: 0755 + +- name: Registry | Templates list + set_fact: + registry_templates: + - { name: registry-ns, file: registry-ns.yml, type: ns } + - { name: registry-sa, file: registry-sa.yml, type: sa } + - { name: registry-svc, file: registry-svc.yml, type: svc } + - { name: registry-secrets, file: registry-secrets.yml, type: secrets } + - { name: registry-cm, file: registry-cm.yml, type: cm } + - { name: registry-rs, file: registry-rs.yml, type: rs } + registry_templates_for_psp: + - { name: registry-psp, file: registry-psp.yml, type: psp } + - { name: registry-cr, file: registry-cr.yml, type: clusterrole } + - { name: registry-crb, file: registry-crb.yml, type: rolebinding } + +- name: Registry | Append extra templates to Registry Templates list for PodSecurityPolicy + set_fact: + registry_templates: "{{ registry_templates[:2] + registry_templates_for_psp + registry_templates[2:] }}" + when: + - podsecuritypolicy_enabled + - registry_namespace != "kube-system" + +- name: Registry | Append nginx ingress templates to Registry Templates list when ingress enabled + set_fact: + registry_templates: "{{ registry_templates + [item] }}" + with_items: + - [{ name: registry-ing, file: registry-ing.yml, type: ing }] + when: ingress_nginx_enabled or ingress_alb_enabled + +- name: Registry | Create manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/registry/{{ item.file }}" + mode: 0644 + with_items: "{{ registry_templates }}" + register: registry_manifests + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Registry | Apply manifests + kube: + name: "{{ item.item.name }}" + namespace: "{{ registry_namespace }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/registry/{{ item.item.file }}" + state: "latest" + with_items: "{{ registry_manifests.results }}" + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Registry | Create PVC manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/registry/{{ item.file }}" + mode: 0644 + with_items: + - { name: registry-pvc, file: registry-pvc.yml, type: pvc } + register: registry_manifests + when: + - registry_storage_class != none and registry_storage_class + - registry_disk_size != none and registry_disk_size + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Registry | Apply PVC manifests + kube: + name: "{{ item.item.name }}" + namespace: "{{ registry_namespace }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/registry/{{ item.item.file }}" + state: "latest" + with_items: "{{ registry_manifests.results }}" + when: + - registry_storage_class != none and registry_storage_class + - registry_disk_size != none and registry_disk_size + - inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/kubernetes-apps/registry/templates/registry-cm.yml.j2 b/kubespray/roles/kubernetes-apps/registry/templates/registry-cm.yml.j2 new file mode 100644 index 0000000..b633dfd --- /dev/null +++ b/kubespray/roles/kubernetes-apps/registry/templates/registry-cm.yml.j2 @@ -0,0 +1,10 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: registry-config + namespace: {{ registry_namespace }} +{% if registry_config %} +data: + config.yml: |- + {{ registry_config | to_yaml(indent=2, width=1337) | indent(width=4) }} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/registry/templates/registry-cr.yml.j2 b/kubespray/roles/kubernetes-apps/registry/templates/registry-cr.yml.j2 new file mode 100644 index 0000000..45f3fc4 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/registry/templates/registry-cr.yml.j2 @@ -0,0 +1,15 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: psp:registry + namespace: {{ registry_namespace }} +rules: + - apiGroups: + - policy + resourceNames: + - registry + resources: + - podsecuritypolicies + verbs: + - use diff --git a/kubespray/roles/kubernetes-apps/registry/templates/registry-crb.yml.j2 b/kubespray/roles/kubernetes-apps/registry/templates/registry-crb.yml.j2 new file mode 100644 index 0000000..8589420 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/registry/templates/registry-crb.yml.j2 @@ -0,0 +1,13 @@ +kind: RoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: psp:registry + namespace: {{ registry_namespace }} +subjects: + - kind: ServiceAccount + name: registry + namespace: {{ registry_namespace }} +roleRef: + kind: ClusterRole + name: psp:registry + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/registry/templates/registry-ing.yml.j2 b/kubespray/roles/kubernetes-apps/registry/templates/registry-ing.yml.j2 new file mode 100644 index 0000000..29dfbba --- /dev/null +++ b/kubespray/roles/kubernetes-apps/registry/templates/registry-ing.yml.j2 @@ -0,0 +1,27 @@ +apiVersion: networking.k8s.io/v1 +kind: Ingress +metadata: + name: registry + namespace: {{ registry_namespace }} +{% if registry_ingress_annotations %} + annotations: + {{ registry_ingress_annotations | to_nice_yaml(indent=2, width=1337) | indent(width=4) }} +{% endif %} +spec: +{% if registry_ingress_tls_secret %} + tls: + - hosts: + - {{ registry_ingress_host }} + secretName: {{ registry_ingress_tls_secret }} +{% endif %} + rules: + - host: {{ registry_ingress_host }} + http: + paths: + - path: / + pathType: Prefix + backend: + service: + name: registry + port: + number: {{ registry_port }} diff --git a/kubespray/roles/kubernetes-apps/registry/templates/registry-ns.yml.j2 b/kubespray/roles/kubernetes-apps/registry/templates/registry-ns.yml.j2 new file mode 100644 index 0000000..c224337 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/registry/templates/registry-ns.yml.j2 @@ -0,0 +1,7 @@ +--- +apiVersion: v1 +kind: Namespace +metadata: + name: {{ registry_namespace }} + labels: + name: {{ registry_namespace }} diff --git a/kubespray/roles/kubernetes-apps/registry/templates/registry-psp.yml.j2 b/kubespray/roles/kubernetes-apps/registry/templates/registry-psp.yml.j2 new file mode 100644 index 0000000..b04d8c2 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/registry/templates/registry-psp.yml.j2 @@ -0,0 +1,44 @@ +--- +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: registry + annotations: + seccomp.security.alpha.kubernetes.io/defaultProfileName: 'runtime/default' + seccomp.security.alpha.kubernetes.io/allowedProfileNames: 'runtime/default' +{% if apparmor_enabled %} + apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + apparmor.security.beta.kubernetes.io/allowedProfileNames: 'runtime/default' +{% endif %} + labels: + addonmanager.kubernetes.io/mode: Reconcile +spec: + privileged: false + allowPrivilegeEscalation: false + requiredDropCapabilities: + - ALL + volumes: + - 'configMap' + - 'emptyDir' + - 'projected' + - 'secret' + - 'downwardAPI' + - 'persistentVolumeClaim' + hostNetwork: false + hostIPC: false + hostPID: false + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + readOnlyRootFilesystem: false diff --git a/kubespray/roles/kubernetes-apps/registry/templates/registry-pvc.yml.j2 b/kubespray/roles/kubernetes-apps/registry/templates/registry-pvc.yml.j2 new file mode 100644 index 0000000..dc3fa5a --- /dev/null +++ b/kubespray/roles/kubernetes-apps/registry/templates/registry-pvc.yml.j2 @@ -0,0 +1,15 @@ +--- +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + name: registry-pvc + namespace: {{ registry_namespace }} + labels: + addonmanager.kubernetes.io/mode: Reconcile +spec: + accessModes: + - {{ registry_storage_access_mode }} + storageClassName: {{ registry_storage_class }} + resources: + requests: + storage: {{ registry_disk_size }} diff --git a/kubespray/roles/kubernetes-apps/registry/templates/registry-rs.yml.j2 b/kubespray/roles/kubernetes-apps/registry/templates/registry-rs.yml.j2 new file mode 100644 index 0000000..3b51684 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/registry/templates/registry-rs.yml.j2 @@ -0,0 +1,115 @@ +--- +apiVersion: apps/v1 +kind: ReplicaSet +metadata: + name: registry + namespace: {{ registry_namespace }} + labels: + k8s-app: registry + version: v{{ registry_image_tag }} + addonmanager.kubernetes.io/mode: Reconcile +spec: +{% if registry_storage_class != "" and registry_storage_access_mode == "ReadWriteMany" %} + replicas: {{ registry_replica_count }} +{% else %} + replicas: 1 +{% endif %} + selector: + matchLabels: + k8s-app: registry + version: v{{ registry_image_tag }} + template: + metadata: + labels: + k8s-app: registry + version: v{{ registry_image_tag }} + spec: + priorityClassName: {% if registry_namespace == 'kube-system' %}system-cluster-critical{% else %}k8s-cluster-critical{% endif %}{{ '' }} + serviceAccountName: registry + securityContext: + fsGroup: 1000 + runAsUser: 1000 + containers: + - name: registry + image: {{ registry_image_repo }}:{{ registry_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + command: + - /bin/registry + - serve + - /etc/docker/registry/config.yml + env: + - name: REGISTRY_HTTP_ADDR + value: :{{ registry_port }} + - name: REGISTRY_STORAGE_FILESYSTEM_ROOTDIRECTORY + value: /var/lib/registry +{% if registry_htpasswd != "" %} + - name: REGISTRY_AUTH + value: "htpasswd" + - name: REGISTRY_AUTH_HTPASSWD_REALM + value: "Registry Realm" + - name: REGISTRY_AUTH_HTPASSWD_PATH + value: "/auth/htpasswd" +{% endif %} +{% if registry_tls_secret != "" %} + - name: REGISTRY_HTTP_TLS_CERTIFICATE + value: /etc/ssl/docker/tls.crt + - name: REGISTRY_HTTP_TLS_KEY + value: /etc/ssl/docker/tls.key +{% endif %} + volumeMounts: + - name: registry-pvc + mountPath: /var/lib/registry + - name: registry-config + mountPath: /etc/docker/registry +{% if registry_htpasswd != "" %} + - name: auth + mountPath: /auth + readOnly: true +{% endif %} +{% if registry_tls_secret != "" %} + - name: tls-cert + mountPath: /etc/ssl/docker + readOnly: true +{% endif %} + ports: + - containerPort: {{ registry_port }} + name: registry + protocol: TCP + livenessProbe: + httpGet: +{% if registry_tls_secret != "" %} + scheme: HTTPS +{% endif %} + path: / + port: {{ registry_port }} + readinessProbe: + httpGet: +{% if registry_tls_secret != "" %} + scheme: HTTPS +{% endif %} + path: / + port: {{ registry_port }} + volumes: + - name: registry-pvc +{% if registry_storage_class != "" %} + persistentVolumeClaim: + claimName: registry-pvc +{% else %} + emptyDir: {} +{% endif %} + - name: registry-config + configMap: + name: registry-config +{% if registry_htpasswd != "" %} + - name: auth + secret: + secretName: registry-secret + items: + - key: htpasswd + path: htpasswd +{% endif %} +{% if registry_tls_secret != "" %} + - name: tls-cert + secret: + secretName: {{ registry_tls_secret }} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/registry/templates/registry-sa.yml.j2 b/kubespray/roles/kubernetes-apps/registry/templates/registry-sa.yml.j2 new file mode 100644 index 0000000..20f9515 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/registry/templates/registry-sa.yml.j2 @@ -0,0 +1,5 @@ +apiVersion: v1 +kind: ServiceAccount +metadata: + name: registry + namespace: {{ registry_namespace }} diff --git a/kubespray/roles/kubernetes-apps/registry/templates/registry-secrets.yml.j2 b/kubespray/roles/kubernetes-apps/registry/templates/registry-secrets.yml.j2 new file mode 100644 index 0000000..80727d2 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/registry/templates/registry-secrets.yml.j2 @@ -0,0 +1,10 @@ +apiVersion: v1 +kind: Secret +metadata: + name: registry-secret + namespace: {{ registry_namespace }} +type: Opaque +data: +{% if registry_htpasswd != "" %} + htpasswd: {{ registry_htpasswd | b64encode }} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/registry/templates/registry-svc.yml.j2 b/kubespray/roles/kubernetes-apps/registry/templates/registry-svc.yml.j2 new file mode 100644 index 0000000..5485aa8 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/registry/templates/registry-svc.yml.j2 @@ -0,0 +1,32 @@ +--- +apiVersion: v1 +kind: Service +metadata: + name: registry + namespace: {{ registry_namespace }} + labels: + k8s-app: registry + addonmanager.kubernetes.io/mode: Reconcile + kubernetes.io/name: "KubeRegistry" +{% if registry_service_annotations %} + annotations: + {{ registry_service_annotations | to_nice_yaml(indent=2, width=1337) | indent(width=4) }} +{% endif %} +spec: + selector: + k8s-app: registry + type: {{ registry_service_type }} +{% if registry_service_type == "ClusterIP" and registry_service_cluster_ip != "" %} + clusterIP: {{ registry_service_cluster_ip }} +{% endif %} +{% if registry_service_type == "LoadBalancer" and registry_service_loadbalancer_ip != "" %} + loadBalancerIP: {{ registry_service_loadbalancer_ip }} +{% endif %} + ports: + - name: registry + port: {{ registry_port }} + protocol: TCP + targetPort: {{ registry_port }} +{% if registry_service_type == "NodePort" and registry_service_nodeport != "" %} + nodePort: {{ registry_service_nodeport }} +{% endif %} diff --git a/kubespray/roles/kubernetes-apps/snapshots/cinder-csi/defaults/main.yml b/kubespray/roles/kubernetes-apps/snapshots/cinder-csi/defaults/main.yml new file mode 100644 index 0000000..1186d98 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/snapshots/cinder-csi/defaults/main.yml @@ -0,0 +1,6 @@ +--- +snapshot_classes: + - name: cinder-csi-snapshot + is_default: false + force_create: true + deletionPolicy: Delete diff --git a/kubespray/roles/kubernetes-apps/snapshots/cinder-csi/tasks/main.yml b/kubespray/roles/kubernetes-apps/snapshots/cinder-csi/tasks/main.yml new file mode 100644 index 0000000..7e9116f --- /dev/null +++ b/kubespray/roles/kubernetes-apps/snapshots/cinder-csi/tasks/main.yml @@ -0,0 +1,18 @@ +--- +- name: Kubernetes Snapshots | Copy Cinder CSI Snapshot Class template + template: + src: "cinder-csi-snapshot-class.yml.j2" + dest: "{{ kube_config_dir }}/cinder-csi-snapshot-class.yml" + mode: 0644 + register: manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kubernetes Snapshots | Add Cinder CSI Snapshot Class + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/cinder-csi-snapshot-class.yml" + state: "latest" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - manifests.changed diff --git a/kubespray/roles/kubernetes-apps/snapshots/cinder-csi/templates/cinder-csi-snapshot-class.yml.j2 b/kubespray/roles/kubernetes-apps/snapshots/cinder-csi/templates/cinder-csi-snapshot-class.yml.j2 new file mode 100644 index 0000000..86c73cc --- /dev/null +++ b/kubespray/roles/kubernetes-apps/snapshots/cinder-csi/templates/cinder-csi-snapshot-class.yml.j2 @@ -0,0 +1,13 @@ +{% for class in snapshot_classes %} +--- +kind: VolumeSnapshotClass +apiVersion: snapshot.storage.k8s.io/v1beta1 +metadata: + name: "{{ class.name }}" + annotations: + storageclass.kubernetes.io/is-default-class: "{{ class.is_default | default(false) | ternary("true","false") }}" +driver: cinder.csi.openstack.org +deletionPolicy: "{{ class.deletionPolicy | default("Delete") }}" +parameters: + force-create: "{{ class.force_create }}" +{% endfor %} diff --git a/kubespray/roles/kubernetes-apps/snapshots/meta/main.yml b/kubespray/roles/kubernetes-apps/snapshots/meta/main.yml new file mode 100644 index 0000000..0eed56c --- /dev/null +++ b/kubespray/roles/kubernetes-apps/snapshots/meta/main.yml @@ -0,0 +1,14 @@ +--- +dependencies: + - role: kubernetes-apps/snapshots/snapshot-controller + when: + - cinder_csi_enabled or csi_snapshot_controller_enabled + tags: + - snapshot-controller + + - role: kubernetes-apps/snapshots/cinder-csi + when: + - cinder_csi_enabled + tags: + - snapshot + - cinder-csi-driver diff --git a/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/defaults/main.yml b/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/defaults/main.yml new file mode 100644 index 0000000..c72dfb2 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/defaults/main.yml @@ -0,0 +1,3 @@ +--- +snapshot_controller_replicas: 1 +snapshot_controller_namespace: kube-system diff --git a/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/tasks/main.yml b/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/tasks/main.yml new file mode 100644 index 0000000..e6da292 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/tasks/main.yml @@ -0,0 +1,39 @@ +--- +- name: Check if snapshot namespace exists + register: snapshot_namespace_exists + kube: + kubectl: "{{ bin_dir }}/kubectl" + name: "{{ snapshot_controller_namespace }}" + resource: "namespace" + state: "exists" + when: inventory_hostname == groups['kube_control_plane'][0] + tags: snapshot-controller + +- name: Snapshot Controller | Generate Manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: snapshot-ns, file: snapshot-ns.yml, apply: not snapshot_namespace_exists} + - {name: rbac-snapshot-controller, file: rbac-snapshot-controller.yml} + - {name: snapshot-controller, file: snapshot-controller.yml} + register: snapshot_controller_manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + - item.apply | default(True) | bool + tags: snapshot-controller + +- name: Snapshot Controller | Apply Manifests + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ snapshot_controller_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item.file }}" + tags: snapshot-controller diff --git a/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/templates/rbac-snapshot-controller.yml.j2 b/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/templates/rbac-snapshot-controller.yml.j2 new file mode 100644 index 0000000..9413376 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/templates/rbac-snapshot-controller.yml.j2 @@ -0,0 +1,85 @@ +# RBAC file for the snapshot controller. +# +# The snapshot controller implements the control loop for CSI snapshot functionality. +# It should be installed as part of the base Kubernetes distribution in an appropriate +# namespace for components implementing base system functionality. For installing with +# Vanilla Kubernetes, kube-system makes sense for the namespace. + +apiVersion: v1 +kind: ServiceAccount +metadata: + name: snapshot-controller + namespace: {{ snapshot_controller_namespace }} + +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + # rename if there are conflicts + name: snapshot-controller-runner +rules: + - apiGroups: [""] + resources: ["persistentvolumes"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["persistentvolumeclaims"] + verbs: ["get", "list", "watch", "update"] + - apiGroups: ["storage.k8s.io"] + resources: ["storageclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["list", "watch", "create", "update", "patch"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshotcontents"] + verbs: ["create", "get", "list", "watch", "update", "delete"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshots"] + verbs: ["get", "list", "watch", "update"] + - apiGroups: ["snapshot.storage.k8s.io"] + resources: ["volumesnapshots/status"] + verbs: ["update"] + +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: snapshot-controller-role +subjects: + - kind: ServiceAccount + name: snapshot-controller + namespace: {{ snapshot_controller_namespace }} +roleRef: + kind: ClusterRole + # change the name also here if the ClusterRole gets renamed + name: snapshot-controller-runner + apiGroup: rbac.authorization.k8s.io + +--- +kind: Role +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + namespace: {{ snapshot_controller_namespace }} + name: snapshot-controller-leaderelection +rules: +- apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["get", "watch", "list", "delete", "update", "create"] + +--- +kind: RoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: snapshot-controller-leaderelection + namespace: {{ snapshot_controller_namespace }} +subjects: + - kind: ServiceAccount + name: snapshot-controller + namespace: {{ snapshot_controller_namespace }} +roleRef: + kind: Role + name: snapshot-controller-leaderelection + apiGroup: rbac.authorization.k8s.io diff --git a/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/templates/snapshot-controller.yml.j2 b/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/templates/snapshot-controller.yml.j2 new file mode 100644 index 0000000..d17ffb3 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/templates/snapshot-controller.yml.j2 @@ -0,0 +1,40 @@ +# This YAML file shows how to deploy the snapshot controller + +# The snapshot controller implements the control loop for CSI snapshot functionality. +# It should be installed as part of the base Kubernetes distribution in an appropriate +# namespace for components implementing base system functionality. For installing with +# Vanilla Kubernetes, kube-system makes sense for the namespace. + +--- +kind: Deployment +apiVersion: apps/v1 +metadata: + name: snapshot-controller + namespace: {{ snapshot_controller_namespace }} +spec: + replicas: {{ snapshot_controller_replicas }} + selector: + matchLabels: + app: snapshot-controller + # the snapshot controller won't be marked as ready if the v1 CRDs are unavailable + # in #504 the snapshot-controller will exit after around 7.5 seconds if it + # can't find the v1 CRDs so this value should be greater than that + minReadySeconds: 15 + strategy: + rollingUpdate: + maxSurge: 0 + maxUnavailable: 1 + type: RollingUpdate + template: + metadata: + labels: + app: snapshot-controller + spec: + serviceAccount: snapshot-controller + containers: + - name: snapshot-controller + image: {{ snapshot_controller_image_repo }}:{{ snapshot_controller_image_tag }} + args: + - "--v=5" + - "--leader-election=false" + imagePullPolicy: {{ k8s_image_pull_policy }} diff --git a/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/templates/snapshot-ns.yml.j2 b/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/templates/snapshot-ns.yml.j2 new file mode 100644 index 0000000..bb30d60 --- /dev/null +++ b/kubespray/roles/kubernetes-apps/snapshots/snapshot-controller/templates/snapshot-ns.yml.j2 @@ -0,0 +1,7 @@ +--- +apiVersion: v1 +kind: Namespace +metadata: + name: {{ snapshot_controller_namespace }} + labels: + name: {{ snapshot_controller_namespace }} diff --git a/kubespray/roles/kubernetes/client/defaults/main.yml b/kubespray/roles/kubernetes/client/defaults/main.yml new file mode 100644 index 0000000..83506a4 --- /dev/null +++ b/kubespray/roles/kubernetes/client/defaults/main.yml @@ -0,0 +1,8 @@ +--- +kubeconfig_localhost: false +kubeconfig_localhost_ansible_host: false +kubectl_localhost: false +artifacts_dir: "{{ inventory_dir }}/artifacts" + +kube_config_dir: "/etc/kubernetes" +kube_apiserver_port: "6443" diff --git a/kubespray/roles/kubernetes/client/tasks/main.yml b/kubespray/roles/kubernetes/client/tasks/main.yml new file mode 100644 index 0000000..e619761 --- /dev/null +++ b/kubespray/roles/kubernetes/client/tasks/main.yml @@ -0,0 +1,114 @@ +--- +- name: Set external kube-apiserver endpoint + set_fact: + # noqa: jinja[spacing] + external_apiserver_address: >- + {%- if loadbalancer_apiserver is defined and loadbalancer_apiserver.address is defined -%} + {{ loadbalancer_apiserver.address }} + {%- elif kubeconfig_localhost_ansible_host is defined and kubeconfig_localhost_ansible_host -%} + {{ hostvars[groups['kube_control_plane'][0]].ansible_host }} + {%- else -%} + {{ kube_apiserver_access_address }} + {%- endif -%} + # noqa: jinja[spacing] + external_apiserver_port: >- + {%- if loadbalancer_apiserver is defined and loadbalancer_apiserver.address is defined and loadbalancer_apiserver.port is defined -%} + {{ loadbalancer_apiserver.port | default(kube_apiserver_port) }} + {%- else -%} + {{ kube_apiserver_port }} + {%- endif -%} + tags: + - facts + +- name: Create kube config dir for current/ansible become user + file: + path: "{{ ansible_env.HOME | default('/root') }}/.kube" + mode: "0700" + state: directory + +- name: Copy admin kubeconfig to current/ansible become user home + copy: + src: "{{ kube_config_dir }}/admin.conf" + dest: "{{ ansible_env.HOME | default('/root') }}/.kube/config" + remote_src: yes + mode: "0600" + backup: yes + +- name: Create kube artifacts dir + file: + path: "{{ artifacts_dir }}" + mode: "0750" + state: directory + delegate_to: localhost + connection: local + become: no + run_once: yes + when: kubeconfig_localhost + +- name: Wait for k8s apiserver + wait_for: + host: "{{ kube_apiserver_access_address }}" + port: "{{ kube_apiserver_port }}" + timeout: 180 + +- name: Get admin kubeconfig from remote host + slurp: + src: "{{ kube_config_dir }}/admin.conf" + run_once: yes + register: raw_admin_kubeconfig + when: kubeconfig_localhost + +- name: Convert kubeconfig to YAML + set_fact: + admin_kubeconfig: "{{ raw_admin_kubeconfig.content | b64decode | from_yaml }}" + when: kubeconfig_localhost + +- name: Override username in kubeconfig + set_fact: + final_admin_kubeconfig: "{{ admin_kubeconfig | combine(override_cluster_name, recursive=true) | combine(override_context, recursive=true) | combine(override_user, recursive=true) }}" + vars: + cluster_infos: "{{ admin_kubeconfig['clusters'][0]['cluster'] }}" + user_certs: "{{ admin_kubeconfig['users'][0]['user'] }}" + username: "kubernetes-admin-{{ cluster_name }}" + context: "kubernetes-admin-{{ cluster_name }}@{{ cluster_name }}" + override_cluster_name: "{{ {'clusters': [{'cluster': (cluster_infos | combine({'server': 'https://' + external_apiserver_address + ':' + (external_apiserver_port | string)})), 'name': cluster_name}]} }}" + override_context: "{{ {'contexts': [{'context': {'user': username, 'cluster': cluster_name}, 'name': context}], 'current-context': context} }}" + override_user: "{{ {'users': [{'name': username, 'user': user_certs}]} }}" + when: kubeconfig_localhost + +- name: Write admin kubeconfig on ansible host + copy: + content: "{{ final_admin_kubeconfig | to_nice_yaml(indent=2) }}" + dest: "{{ artifacts_dir }}/admin.conf" + mode: 0600 + delegate_to: localhost + connection: local + become: no + run_once: yes + when: kubeconfig_localhost + +- name: Copy kubectl binary to ansible host + fetch: + src: "{{ bin_dir }}/kubectl" + dest: "{{ artifacts_dir }}/kubectl" + flat: yes + validate_checksum: no + register: copy_binary_result + until: copy_binary_result is not failed + retries: 20 + become: no + run_once: yes + when: kubectl_localhost + +- name: Create helper script kubectl.sh on ansible host + copy: + content: | + #!/bin/bash + ${BASH_SOURCE%/*}/kubectl --kubeconfig=${BASH_SOURCE%/*}/admin.conf "$@" + dest: "{{ artifacts_dir }}/kubectl.sh" + mode: 0755 + become: no + run_once: yes + delegate_to: localhost + connection: local + when: kubectl_localhost and kubeconfig_localhost diff --git a/kubespray/roles/kubernetes/control-plane/defaults/main/etcd.yml b/kubespray/roles/kubernetes/control-plane/defaults/main/etcd.yml new file mode 100644 index 0000000..344ce9b --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/defaults/main/etcd.yml @@ -0,0 +1,31 @@ +--- +# Set etcd user/group +etcd_owner: etcd + +# Note: This does not set up DNS entries. It simply adds the following DNS +# entries to the certificate +etcd_cert_alt_names: + - "etcd.kube-system.svc.{{ dns_domain }}" + - "etcd.kube-system.svc" + - "etcd.kube-system" + - "etcd" +etcd_cert_alt_ips: [] + +etcd_heartbeat_interval: "250" +etcd_election_timeout: "5000" + +# etcd_snapshot_count: "10000" + +etcd_metrics: "basic" + +## A dictionary of extra environment variables to add to etcd.env, formatted like: +## etcd_extra_vars: +## var1: "value1" +## var2: "value2" +## Note this is different from the etcd role with ETCD_ prfexi, caps, and underscores +etcd_extra_vars: {} + +# etcd_quota_backend_bytes: "2147483648" +# etcd_max_request_bytes: "1572864" + +etcd_compaction_retention: "8" diff --git a/kubespray/roles/kubernetes/control-plane/defaults/main/kube-proxy.yml b/kubespray/roles/kubernetes/control-plane/defaults/main/kube-proxy.yml new file mode 100644 index 0000000..24ebc6c --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/defaults/main/kube-proxy.yml @@ -0,0 +1,114 @@ +--- +# bind address for kube-proxy +kube_proxy_bind_address: '0.0.0.0' + +# acceptContentTypes defines the Accept header sent by clients when connecting to a server, overriding the +# default value of 'application/json'. This field will control all connections to the server used by a particular +# client. +kube_proxy_client_accept_content_types: '' + +# burst allows extra queries to accumulate when a client is exceeding its rate. +kube_proxy_client_burst: 10 + +# contentType is the content type used when sending data to the server from this client. +kube_proxy_client_content_type: application/vnd.kubernetes.protobuf + +# kubeconfig is the path to a KubeConfig file. +# Leave as empty string to generate from other fields +kube_proxy_client_kubeconfig: '' + +# qps controls the number of queries per second allowed for this connection. +kube_proxy_client_qps: 5 + +# How often configuration from the apiserver is refreshed. Must be greater than 0. +kube_proxy_config_sync_period: 15m0s + +### Conntrack +# maxPerCore is the maximum number of NAT connections to track +# per CPU core (0 to leave the limit as-is and ignore min). +kube_proxy_conntrack_max_per_core: 32768 + +# min is the minimum value of connect-tracking records to allocate, +# regardless of conntrackMaxPerCore (set maxPerCore=0 to leave the limit as-is). +kube_proxy_conntrack_min: 131072 + +# tcpCloseWaitTimeout is how long an idle conntrack entry +# in CLOSE_WAIT state will remain in the conntrack +# table. (e.g. '60s'). Must be greater than 0 to set. +kube_proxy_conntrack_tcp_close_wait_timeout: 1h0m0s + +# tcpEstablishedTimeout is how long an idle TCP connection will be kept open +# (e.g. '2s'). Must be greater than 0 to set. +kube_proxy_conntrack_tcp_established_timeout: 24h0m0s + +# Enables profiling via web interface on /debug/pprof handler. +# Profiling handlers will be handled by metrics server. +kube_proxy_enable_profiling: false + +# bind address for kube-proxy health check +kube_proxy_healthz_bind_address: 0.0.0.0:10256 + +# If using the pure iptables proxy, SNAT everything. Note that it breaks any +# policy engine. +kube_proxy_masquerade_all: false + +# If using the pure iptables proxy, the bit of the fwmark space to mark packets requiring SNAT with. +# Must be within the range [0, 31]. +kube_proxy_masquerade_bit: 14 + +# The minimum interval of how often the iptables or ipvs rules can be refreshed as +# endpoints and services change (e.g. '5s', '1m', '2h22m'). +kube_proxy_min_sync_period: 0s + +# The maximum interval of how often iptables or ipvs rules are refreshed (e.g. '5s', '1m', '2h22m'). +# Must be greater than 0. +kube_proxy_sync_period: 30s + +# A comma-separated list of CIDR's which the ipvs proxier should not touch when cleaning up IPVS rules. +kube_proxy_exclude_cidrs: [] + +# The ipvs scheduler type when proxy mode is ipvs +# rr: round-robin +# lc: least connection +# dh: destination hashing +# sh: source hashing +# sed: shortest expected delay +# nq: never queue +kube_proxy_scheduler: rr + +# configure arp_ignore and arp_announce to avoid answering ARP queries from kube-ipvs0 interface +# must be set to true for MetalLB, kube-vip(ARP enabled) to work +kube_proxy_strict_arp: false + +# kube_proxy_tcp_timeout is the timeout value used for idle IPVS TCP sessions. +# The default value is 0, which preserves the current timeout value on the system. +kube_proxy_tcp_timeout: 0s + +# kube_proxy_tcp_fin_timeout is the timeout value used for IPVS TCP sessions after receiving a FIN. +# The default value is 0, which preserves the current timeout value on the system. +kube_proxy_tcp_fin_timeout: 0s + +# kube_proxy_udp_timeout is the timeout value used for IPVS UDP packets. +# The default value is 0, which preserves the current timeout value on the system. +kube_proxy_udp_timeout: 0s + +# The IP address and port for the metrics server to serve on +# (set to 0.0.0.0 for all IPv4 interfaces and `::` for all IPv6 interfaces) +kube_proxy_metrics_bind_address: 127.0.0.1:10249 + +# A string slice of values which specify the addresses to use for NodePorts. +# Values may be valid IP blocks (e.g. 1.2.3.0/24, 1.2.3.4/32). +# The default empty string slice ([]) means to use all local addresses. +kube_proxy_nodeport_addresses: >- + {%- if kube_proxy_nodeport_addresses_cidr is defined -%} + [{{ kube_proxy_nodeport_addresses_cidr }}] + {%- else -%} + [] + {%- endif -%} + +# oom-score-adj value for kube-proxy process. Values must be within the range [-1000, 1000] +kube_proxy_oom_score_adj: -999 + +# portRange is the range of host ports (beginPort-endPort, inclusive) that may be consumed +# in order to proxy service traffic. If unspecified, 0, or (0-0) then ports will be randomly chosen. +kube_proxy_port_range: '' diff --git a/kubespray/roles/kubernetes/control-plane/defaults/main/kube-scheduler.yml b/kubespray/roles/kubernetes/control-plane/defaults/main/kube-scheduler.yml new file mode 100644 index 0000000..e61bcb7 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/defaults/main/kube-scheduler.yml @@ -0,0 +1,33 @@ +--- +# Extra args passed by kubeadm +kube_kubeadm_scheduler_extra_args: {} + +# Associated interface must be reachable by the rest of the cluster, and by +# CLI/web clients. +kube_scheduler_bind_address: 0.0.0.0 + +# ClientConnection options (e.g. Burst, QPS) except from kubeconfig. +kube_scheduler_client_conn_extra_opts: {} + +# Additional KubeSchedulerConfiguration settings (e.g. metricsBindAddress). +kube_scheduler_config_extra_opts: {} + +# List of scheduler extenders (dicts), each holding the values of how to +# communicate with the extender. +kube_scheduler_extenders: [] + +# Leader Election options (e.g. ResourceName, RetryPerion) except from +# LeaseDuration and Renew deadline which are defined in following vars. +kube_scheduler_leader_elect_extra_opts: {} + +# Leader election lease duration +kube_scheduler_leader_elect_lease_duration: 15s + +# Leader election lease timeout +kube_scheduler_leader_elect_renew_deadline: 10s + +# Lisf of scheduling profiles (ditcs) supported by kube-scheduler +kube_scheduler_profiles: [] + +# Extra volume mounts +scheduler_extra_volumes: {} diff --git a/kubespray/roles/kubernetes/control-plane/defaults/main/main.yml b/kubespray/roles/kubernetes/control-plane/defaults/main/main.yml new file mode 100644 index 0000000..2a9eda1 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/defaults/main/main.yml @@ -0,0 +1,230 @@ +--- +# disable upgrade cluster +upgrade_cluster_setup: false + +# By default the external API listens on all interfaces, this can be changed to +# listen on a specific address/interface. +# NOTE: If you specific address/interface and use loadbalancer_apiserver_localhost +# loadbalancer_apiserver_localhost (nginx/haproxy) will deploy on masters on 127.0.0.1:{{ loadbalancer_apiserver_port | default(kube_apiserver_port) }} too. +kube_apiserver_bind_address: 0.0.0.0 + +# A port range to reserve for services with NodePort visibility. +# Inclusive at both ends of the range. +kube_apiserver_node_port_range: "30000-32767" + +# ETCD backend for k8s data +kube_apiserver_storage_backend: etcd3 + +# CIS 1.2.26 +# Validate that the service account token +# in the request is actually present in etcd. +kube_apiserver_service_account_lookup: true + +kube_etcd_cacert_file: ca.pem +kube_etcd_cert_file: node-{{ inventory_hostname }}.pem +kube_etcd_key_file: node-{{ inventory_hostname }}-key.pem + +# Associated interfaces must be reachable by the rest of the cluster, and by +# CLI/web clients. +kube_controller_manager_bind_address: 0.0.0.0 + +# Leader election lease durations and timeouts for controller-manager +kube_controller_manager_leader_elect_lease_duration: 15s +kube_controller_manager_leader_elect_renew_deadline: 10s + +# discovery_timeout modifies the discovery timeout +discovery_timeout: 5m0s + +# Instruct first master to refresh kubeadm token +kubeadm_refresh_token: true + +# Scale down coredns replicas to 0 if not using coredns dns_mode +kubeadm_scale_down_coredns_enabled: true + +# audit support +kubernetes_audit: false +# path to audit log file +audit_log_path: /var/log/audit/kube-apiserver-audit.log +# num days +audit_log_maxage: 30 +# the num of audit logs to retain +audit_log_maxbackups: 10 +# the max size in MB to retain +audit_log_maxsize: 100 +# policy file +audit_policy_file: "{{ kube_config_dir }}/audit-policy/apiserver-audit-policy.yaml" +# custom audit policy rules (to replace the default ones) +# audit_policy_custom_rules: | +# - level: None +# users: [] +# verbs: [] +# resources: [] + +# audit log hostpath +audit_log_name: audit-logs +audit_log_hostpath: /var/log/kubernetes/audit +audit_log_mountpath: "{{ audit_log_path | dirname }}" + +# audit policy hostpath +audit_policy_name: audit-policy +audit_policy_hostpath: "{{ audit_policy_file | dirname }}" +audit_policy_mountpath: "{{ audit_policy_hostpath }}" + +# audit webhook support +kubernetes_audit_webhook: false + +# path to audit webhook config file +audit_webhook_config_file: "{{ kube_config_dir }}/audit-policy/apiserver-audit-webhook-config.yaml" +audit_webhook_server_url: "https://audit.app" +audit_webhook_server_extra_args: {} +audit_webhook_mode: batch +audit_webhook_batch_max_size: 100 +audit_webhook_batch_max_wait: 1s + +kube_controller_node_monitor_grace_period: 40s +kube_controller_node_monitor_period: 5s +kube_controller_terminated_pod_gc_threshold: 12500 +kube_apiserver_request_timeout: "1m0s" +kube_apiserver_pod_eviction_not_ready_timeout_seconds: "300" +kube_apiserver_pod_eviction_unreachable_timeout_seconds: "300" + +# 1.10+ admission plugins +kube_apiserver_enable_admission_plugins: [] + +# enable admission plugins configuration +kube_apiserver_admission_control_config_file: false + +# data structure to configure EventRateLimit admission plugin +# this should have the following structure: +# kube_apiserver_admission_event_rate_limits: +# : +# type: +# qps: +# burst: +# cache_size: +kube_apiserver_admission_event_rate_limits: {} + +kube_pod_security_use_default: false +kube_pod_security_default_enforce: baseline +kube_pod_security_default_enforce_version: "{{ kube_major_version }}" +kube_pod_security_default_audit: restricted +kube_pod_security_default_audit_version: "{{ kube_major_version }}" +kube_pod_security_default_warn: restricted +kube_pod_security_default_warn_version: "{{ kube_major_version }}" +kube_pod_security_exemptions_usernames: [] +kube_pod_security_exemptions_runtime_class_names: [] +kube_pod_security_exemptions_namespaces: + - kube-system + +# 1.10+ list of disabled admission plugins +kube_apiserver_disable_admission_plugins: [] + +# extra runtime config +kube_api_runtime_config: [] + +## Enable/Disable Kube API Server Authentication Methods +kube_token_auth: false +kube_oidc_auth: false + +## Variables for webhook token auth https://kubernetes.io/docs/reference/access-authn-authz/authentication/#webhook-token-authentication +kube_webhook_token_auth: false +kube_webhook_token_auth_url_skip_tls_verify: false +# kube_webhook_token_auth_url: https://... +## base64-encoded string of the webhook's CA certificate +# kube_webhook_token_auth_ca_data: "LS0t..." + +## Variables for webhook token authz https://kubernetes.io/docs/reference/access-authn-authz/webhook/ +# kube_webhook_authorization_url: https://... +kube_webhook_authorization: false +kube_webhook_authorization_url_skip_tls_verify: false + + +## Variables for OpenID Connect Configuration https://kubernetes.io/docs/admin/authentication/ +## To use OpenID you have to deploy additional an OpenID Provider (e.g Dex, Keycloak, ...) + +# kube_oidc_url: https:// ... +# kube_oidc_client_id: kubernetes +## Optional settings for OIDC +# kube_oidc_username_claim: sub +# kube_oidc_username_prefix: 'oidc:' +# kube_oidc_groups_claim: groups +# kube_oidc_groups_prefix: 'oidc:' +# Copy oidc CA file to the following path if needed +# kube_oidc_ca_file: {{ kube_cert_dir }}/ca.pem +# Optionally include a base64-encoded oidc CA cert +# kube_oidc_ca_cert: c3RhY2thYnVzZS5jb20... + +# List of the preferred NodeAddressTypes to use for kubelet connections. +kubelet_preferred_address_types: 'InternalDNS,InternalIP,Hostname,ExternalDNS,ExternalIP' + +## Extra args for k8s components passing by kubeadm +kube_kubeadm_apiserver_extra_args: {} +kube_kubeadm_controller_extra_args: {} + +## Extra control plane host volume mounts +## Example: +# apiserver_extra_volumes: +# - name: name +# hostPath: /host/path +# mountPath: /mount/path +# readOnly: true +apiserver_extra_volumes: {} +controller_manager_extra_volumes: {} + +## Encrypting Secret Data at Rest +kube_encrypt_secret_data: false +kube_encrypt_token: "{{ lookup('password', credentials_dir + '/kube_encrypt_token.creds length=32 chars=ascii_letters,digits') }}" +# Must be either: aescbc, secretbox or aesgcm +kube_encryption_algorithm: "secretbox" +# Which kubernetes resources to encrypt +kube_encryption_resources: [secrets] + +# If non-empty, will use this string as identification instead of the actual hostname +kube_override_hostname: >- + {%- if cloud_provider is defined and cloud_provider in ['aws'] -%} + {%- else -%} + {{ inventory_hostname }} + {%- endif -%} + +secrets_encryption_query: "resources[*].providers[0].{{ kube_encryption_algorithm }}.keys[0].secret" + +## Support tls min version, Possible values: VersionTLS10, VersionTLS11, VersionTLS12, VersionTLS13. +# tls_min_version: "" + +## Support tls cipher suites. +# tls_cipher_suites: +# - TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA +# - TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256 +# - TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 +# - TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA +# - TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 +# - TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305 +# - TLS_ECDHE_ECDSA_WITH_RC4_128_SHA +# - TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256 +# - TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 +# - TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 +# - TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305 +# - TLS_ECDHE_RSA_WITH_RC4_128_SHA +# - TLS_RSA_WITH_3DES_EDE_CBC_SHA +# - TLS_RSA_WITH_AES_128_CBC_SHA +# - TLS_RSA_WITH_AES_128_CBC_SHA256 +# - TLS_RSA_WITH_AES_128_GCM_SHA256 +# - TLS_RSA_WITH_AES_256_CBC_SHA +# - TLS_RSA_WITH_AES_256_GCM_SHA384 +# - TLS_RSA_WITH_RC4_128_SHA + +## Amount of time to retain events. (default 1h0m0s) +event_ttl_duration: "1h0m0s" + +## Automatically renew K8S control plane certificates on first Monday of each month +auto_renew_certificates: false +# First Monday of each month +auto_renew_certificates_systemd_calendar: "{{ 'Mon *-*-1,2,3,4,5,6,7 03:' ~ + groups['kube_control_plane'].index(inventory_hostname) ~ '0:00' }}" +# kubeadm renews all the certificates during control plane upgrade. +# If we have requirement like without renewing certs upgrade the cluster, +# we can opt out from the default behavior by setting kubeadm_upgrade_auto_cert_renewal to false +kubeadm_upgrade_auto_cert_renewal: true diff --git a/kubespray/roles/kubernetes/control-plane/handlers/main.yml b/kubespray/roles/kubernetes/control-plane/handlers/main.yml new file mode 100644 index 0000000..d5f1796 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/handlers/main.yml @@ -0,0 +1,135 @@ +--- +- name: Master | restart kubelet + command: /bin/true + notify: + - Master | reload systemd + - Master | reload kubelet + - Master | wait for master static pods + +- name: Master | wait for master static pods + command: /bin/true + notify: + - Master | wait for the apiserver to be running + - Master | wait for kube-scheduler + - Master | wait for kube-controller-manager + +- name: Master | Restart apiserver + command: /bin/true + notify: + - Master | Remove apiserver container docker + - Master | Remove apiserver container containerd/crio + - Master | wait for the apiserver to be running + +- name: Master | Restart kube-scheduler + command: /bin/true + notify: + - Master | Remove scheduler container docker + - Master | Remove scheduler container containerd/crio + - Master | wait for kube-scheduler + +- name: Master | Restart kube-controller-manager + command: /bin/true + notify: + - Master | Remove controller manager container docker + - Master | Remove controller manager container containerd/crio + - Master | wait for kube-controller-manager + +- name: Master | reload systemd + systemd: + daemon_reload: true + +- name: Master | reload kubelet + service: + name: kubelet + state: restarted + +- name: Master | Remove apiserver container docker + shell: "set -o pipefail && docker ps -af name=k8s_kube-apiserver* -q | xargs --no-run-if-empty docker rm -f" + args: + executable: /bin/bash + register: remove_apiserver_container + retries: 10 + until: remove_apiserver_container.rc == 0 + delay: 1 + when: container_manager == "docker" + +- name: Master | Remove apiserver container containerd/crio + shell: "set -o pipefail && {{ bin_dir }}/crictl pods --name kube-apiserver* -q | xargs -I% --no-run-if-empty bash -c '{{ bin_dir }}/crictl stopp % && {{ bin_dir }}/crictl rmp %'" + args: + executable: /bin/bash + register: remove_apiserver_container + retries: 10 + until: remove_apiserver_container.rc == 0 + delay: 1 + when: container_manager in ['containerd', 'crio'] + +- name: Master | Remove scheduler container docker + shell: "set -o pipefail && {{ docker_bin_dir }}/docker ps -af name=k8s_kube-scheduler* -q | xargs --no-run-if-empty {{ docker_bin_dir }}/docker rm -f" + args: + executable: /bin/bash + register: remove_scheduler_container + retries: 10 + until: remove_scheduler_container.rc == 0 + delay: 1 + when: container_manager == "docker" + +- name: Master | Remove scheduler container containerd/crio + shell: "set -o pipefail && {{ bin_dir }}/crictl pods --name kube-scheduler* -q | xargs -I% --no-run-if-empty bash -c '{{ bin_dir }}/crictl stopp % && {{ bin_dir }}/crictl rmp %'" + args: + executable: /bin/bash + register: remove_scheduler_container + retries: 10 + until: remove_scheduler_container.rc == 0 + delay: 1 + when: container_manager in ['containerd', 'crio'] + +- name: Master | Remove controller manager container docker + shell: "set -o pipefail && {{ docker_bin_dir }}/docker ps -af name=k8s_kube-controller-manager* -q | xargs --no-run-if-empty {{ docker_bin_dir }}/docker rm -f" + args: + executable: /bin/bash + register: remove_cm_container + retries: 10 + until: remove_cm_container.rc == 0 + delay: 1 + when: container_manager == "docker" + +- name: Master | Remove controller manager container containerd/crio + shell: "set -o pipefail && {{ bin_dir }}/crictl pods --name kube-controller-manager* -q | xargs -I% --no-run-if-empty bash -c '{{ bin_dir }}/crictl stopp % && {{ bin_dir }}/crictl rmp %'" + args: + executable: /bin/bash + register: remove_cm_container + retries: 10 + until: remove_cm_container.rc == 0 + delay: 1 + when: container_manager in ['containerd', 'crio'] + +- name: Master | wait for kube-scheduler + vars: + endpoint: "{{ kube_scheduler_bind_address if kube_scheduler_bind_address != '0.0.0.0' else 'localhost' }}" + uri: + url: https://{{ endpoint }}:10259/healthz + validate_certs: no + register: scheduler_result + until: scheduler_result.status == 200 + retries: 60 + delay: 1 + +- name: Master | wait for kube-controller-manager + vars: + endpoint: "{{ kube_controller_manager_bind_address if kube_controller_manager_bind_address != '0.0.0.0' else 'localhost' }}" + uri: + url: https://{{ endpoint }}:10257/healthz + validate_certs: no + register: controller_manager_result + until: controller_manager_result.status == 200 + retries: 60 + delay: 1 + +- name: Master | wait for the apiserver to be running + uri: + url: "{{ kube_apiserver_endpoint }}/healthz" + validate_certs: no + register: result + until: result.status == 200 + retries: 60 + delay: 1 diff --git a/kubespray/roles/kubernetes/control-plane/meta/main.yml b/kubespray/roles/kubernetes/control-plane/meta/main.yml new file mode 100644 index 0000000..2657006 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/meta/main.yml @@ -0,0 +1,11 @@ +--- +dependencies: + - role: kubernetes/tokens + when: kube_token_auth + tags: + - k8s-secrets + - role: adduser + user: "{{ addusers.etcd }}" + when: + - etcd_deployment_type == "kubeadm" + - not (ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk", "ClearLinux"] or is_fedora_coreos) diff --git a/kubespray/roles/kubernetes/control-plane/tasks/define-first-kube-control.yml b/kubespray/roles/kubernetes/control-plane/tasks/define-first-kube-control.yml new file mode 100644 index 0000000..64e2de7 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/tasks/define-first-kube-control.yml @@ -0,0 +1,19 @@ +--- + +- name: Check which kube-control nodes are already members of the cluster + command: "{{ bin_dir }}/kubectl get nodes --selector=node-role.kubernetes.io/control-plane -o json" + register: kube_control_planes_raw + ignore_errors: yes + changed_when: false + +- name: Set fact joined_control_panes + set_fact: + joined_control_planes: "{{ ((kube_control_planes_raw.stdout | from_json)['items']) | default([]) | map(attribute='metadata') | map(attribute='name') | list }}" + delegate_to: item + loop: "{{ groups['kube_control_plane'] }}" + when: kube_control_planes_raw is succeeded + run_once: yes + +- name: Set fact first_kube_control_plane + set_fact: + first_kube_control_plane: "{{ joined_control_planes | default([]) | first | default(groups['kube_control_plane'] | first) }}" diff --git a/kubespray/roles/kubernetes/control-plane/tasks/encrypt-at-rest.yml b/kubespray/roles/kubernetes/control-plane/tasks/encrypt-at-rest.yml new file mode 100644 index 0000000..209e4c7 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/tasks/encrypt-at-rest.yml @@ -0,0 +1,40 @@ +--- +- name: Check if secret for encrypting data at rest already exist + stat: + path: "{{ kube_cert_dir }}/secrets_encryption.yaml" + get_attributes: no + get_checksum: no + get_mime: no + register: secrets_encryption_file + +- name: Slurp secrets_encryption file if it exists + slurp: + src: "{{ kube_cert_dir }}/secrets_encryption.yaml" + register: secret_file_encoded + when: secrets_encryption_file.stat.exists + +- name: Base 64 Decode slurped secrets_encryption.yaml file + set_fact: + secret_file_decoded: "{{ secret_file_encoded['content'] | b64decode | from_yaml }}" + when: secrets_encryption_file.stat.exists + +- name: Extract secret value from secrets_encryption.yaml + set_fact: + kube_encrypt_token_extracted: "{{ secret_file_decoded | json_query(secrets_encryption_query) | first | b64decode }}" + when: secrets_encryption_file.stat.exists + +- name: Set kube_encrypt_token across master nodes + set_fact: + kube_encrypt_token: "{{ kube_encrypt_token_extracted }}" + delegate_to: "{{ item }}" + delegate_facts: true + with_inventory_hostnames: kube_control_plane + when: kube_encrypt_token_extracted is defined + +- name: Write secrets for encrypting secret data at rest + template: + src: secrets_encryption.yaml.j2 + dest: "{{ kube_cert_dir }}/secrets_encryption.yaml" + owner: root + group: "{{ kube_cert_group }}" + mode: 0640 diff --git a/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-backup.yml b/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-backup.yml new file mode 100644 index 0000000..36bb627 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-backup.yml @@ -0,0 +1,28 @@ +--- +- name: Backup old certs and keys + copy: + src: "{{ kube_cert_dir }}/{{ item }}" + dest: "{{ kube_cert_dir }}/{{ item }}.old" + mode: preserve + remote_src: yes + with_items: + - apiserver.crt + - apiserver.key + - apiserver-kubelet-client.crt + - apiserver-kubelet-client.key + - front-proxy-client.crt + - front-proxy-client.key + ignore_errors: true # noqa ignore-errors + +- name: Backup old confs + copy: + src: "{{ kube_config_dir }}/{{ item }}" + dest: "{{ kube_config_dir }}/{{ item }}.old" + mode: preserve + remote_src: yes + with_items: + - admin.conf + - controller-manager.conf + - kubelet.conf + - scheduler.conf + ignore_errors: true # noqa ignore-errors diff --git a/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-etcd.yml b/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-etcd.yml new file mode 100644 index 0000000..9de55c5 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-etcd.yml @@ -0,0 +1,29 @@ +--- +- name: Calculate etcd cert serial + command: "openssl x509 -in {{ kube_cert_dir }}/apiserver-etcd-client.crt -noout -serial" + register: "etcd_client_cert_serial_result" + changed_when: false + tags: + - network + +- name: Set etcd_client_cert_serial + set_fact: + etcd_client_cert_serial: "{{ etcd_client_cert_serial_result.stdout.split('=')[1] }}" + tags: + - network + +- name: Ensure etcdctl and etcdutl script is installed + import_role: + name: etcdctl_etcdutl + when: etcd_deployment_type == "kubeadm" + tags: + - etcdctl + - etcdutl + +- name: Set ownership for etcd data directory + file: + path: "{{ etcd_data_dir }}" + owner: "{{ etcd_owner }}" + group: "{{ etcd_owner }}" + mode: 0700 + when: etcd_deployment_type == "kubeadm" diff --git a/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-fix-apiserver.yml b/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-fix-apiserver.yml new file mode 100644 index 0000000..8f2f38e --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-fix-apiserver.yml @@ -0,0 +1,24 @@ +--- + +- name: Update server field in component kubeconfigs + lineinfile: + dest: "{{ kube_config_dir }}/{{ item }}" + regexp: '^ server: https' + line: ' server: {{ kube_apiserver_endpoint }}' + backup: yes + with_items: + - admin.conf + - controller-manager.conf + - kubelet.conf + - scheduler.conf + notify: + - "Master | Restart kube-controller-manager" + - "Master | Restart kube-scheduler" + - "Master | reload kubelet" + +- name: Update etcd-servers for apiserver + lineinfile: + dest: "{{ kube_config_dir }}/manifests/kube-apiserver.yaml" + regexp: '^ - --etcd-servers=' + line: ' - --etcd-servers={{ etcd_access_addresses }}' + when: etcd_deployment_type != "kubeadm" diff --git a/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-secondary.yml b/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-secondary.yml new file mode 100644 index 0000000..f3fd207 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-secondary.yml @@ -0,0 +1,80 @@ +--- +- name: Set kubeadm_discovery_address + set_fact: + # noqa: jinja[spacing] + kubeadm_discovery_address: >- + {%- if "127.0.0.1" in kube_apiserver_endpoint or "localhost" in kube_apiserver_endpoint -%} + {{ first_kube_control_plane_address }}:{{ kube_apiserver_port }} + {%- else -%} + {{ kube_apiserver_endpoint | regex_replace('https://', '') }} + {%- endif %} + tags: + - facts + +- name: Upload certificates so they are fresh and not expired + command: >- + {{ bin_dir }}/kubeadm init phase + --config {{ kube_config_dir }}/kubeadm-config.yaml + upload-certs + --upload-certs + register: kubeadm_upload_cert + when: + - inventory_hostname == first_kube_control_plane + - not kube_external_ca_mode + +- name: Parse certificate key if not set + set_fact: + kubeadm_certificate_key: "{{ hostvars[groups['kube_control_plane'][0]]['kubeadm_upload_cert'].stdout_lines[-1] | trim }}" + run_once: yes + when: + - hostvars[groups['kube_control_plane'][0]]['kubeadm_upload_cert'] is defined + - hostvars[groups['kube_control_plane'][0]]['kubeadm_upload_cert'] is not skipped + +- name: Create kubeadm ControlPlane config + template: + src: "kubeadm-controlplane.{{ kubeadmConfig_api_version }}.yaml.j2" + dest: "{{ kube_config_dir }}/kubeadm-controlplane.yaml" + mode: 0640 + backup: yes + when: + - inventory_hostname != first_kube_control_plane + - not kubeadm_already_run.stat.exists + +- name: Wait for k8s apiserver + wait_for: + host: "{{ kubeadm_discovery_address.split(':')[0] }}" + port: "{{ kubeadm_discovery_address.split(':')[1] }}" + timeout: 180 + + +- name: Check already run + debug: + msg: "{{ kubeadm_already_run.stat.exists }}" + +- name: Reset cert directory + shell: >- + if [ -f /etc/kubernetes/manifests/kube-apiserver.yaml ]; then + {{ bin_dir }}/kubeadm reset -f --cert-dir {{ kube_cert_dir }}; + fi + environment: + PATH: "{{ bin_dir }}:{{ ansible_env.PATH }}" + when: + - inventory_hostname != first_kube_control_plane + - kubeadm_already_run is not defined or not kubeadm_already_run.stat.exists + - not kube_external_ca_mode + +- name: Joining control plane node to the cluster. + command: >- + {{ bin_dir }}/kubeadm join + --config {{ kube_config_dir }}/kubeadm-controlplane.yaml + --ignore-preflight-errors=all + --skip-phases={{ kubeadm_join_phases_skip | join(',') }} + environment: + PATH: "{{ bin_dir }}:{{ ansible_env.PATH }}" + register: kubeadm_join_control_plane + retries: 3 + throttle: 1 + until: kubeadm_join_control_plane is succeeded + when: + - inventory_hostname != first_kube_control_plane + - kubeadm_already_run is not defined or not kubeadm_already_run.stat.exists diff --git a/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-setup.yml b/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-setup.yml new file mode 100644 index 0000000..fade43c --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-setup.yml @@ -0,0 +1,249 @@ +--- +- name: Install OIDC certificate + copy: + content: "{{ kube_oidc_ca_cert | b64decode }}" + dest: "{{ kube_oidc_ca_file }}" + owner: root + group: root + mode: "0644" + when: + - kube_oidc_auth + - kube_oidc_ca_cert is defined + +- name: Kubeadm | Check if kubeadm has already run + stat: + path: "/var/lib/kubelet/config.yaml" + get_attributes: no + get_checksum: no + get_mime: no + register: kubeadm_already_run + +- name: Kubeadm | Backup kubeadm certs / kubeconfig + import_tasks: kubeadm-backup.yml + when: + - kubeadm_already_run.stat.exists + +- name: Kubeadm | aggregate all SANs + set_fact: + apiserver_sans: "{{ (sans_base + groups['kube_control_plane'] + sans_lb + sans_lb_ip + sans_supp + sans_access_ip + sans_ip + sans_address + sans_override + sans_hostname + sans_fqdn + sans_kube_vip_address) | unique }}" + vars: + sans_base: + - "kubernetes" + - "kubernetes.default" + - "kubernetes.default.svc" + - "kubernetes.default.svc.{{ dns_domain }}" + - "{{ kube_apiserver_ip }}" + - "localhost" + - "127.0.0.1" + sans_lb: "{{ [apiserver_loadbalancer_domain_name] if apiserver_loadbalancer_domain_name is defined else [] }}" + sans_lb_ip: "{{ [loadbalancer_apiserver.address] if loadbalancer_apiserver is defined and loadbalancer_apiserver.address is defined else [] }}" + sans_supp: "{{ supplementary_addresses_in_ssl_keys if supplementary_addresses_in_ssl_keys is defined else [] }}" + sans_access_ip: "{{ groups['kube_control_plane'] | map('extract', hostvars, 'access_ip') | list | select('defined') | list }}" + sans_ip: "{{ groups['kube_control_plane'] | map('extract', hostvars, 'ip') | list | select('defined') | list }}" + sans_address: "{{ groups['kube_control_plane'] | map('extract', hostvars, ['ansible_default_ipv4', 'address']) | list | select('defined') | list }}" + sans_override: "{{ [kube_override_hostname] if kube_override_hostname else [] }}" + sans_hostname: "{{ groups['kube_control_plane'] | map('extract', hostvars, ['ansible_hostname']) | list | select('defined') | list }}" + sans_fqdn: "{{ groups['kube_control_plane'] | map('extract', hostvars, ['ansible_fqdn']) | list | select('defined') | list }}" + sans_kube_vip_address: "{{ [kube_vip_address] if kube_vip_address is defined and kube_vip_address else [] }}" + tags: facts + +- name: Create audit-policy directory + file: + path: "{{ audit_policy_file | dirname }}" + state: directory + mode: 0640 + when: kubernetes_audit | default(false) or kubernetes_audit_webhook | default(false) + +- name: Write api audit policy yaml + template: + src: apiserver-audit-policy.yaml.j2 + dest: "{{ audit_policy_file }}" + mode: 0640 + when: kubernetes_audit | default(false) or kubernetes_audit_webhook | default(false) + +- name: Write api audit webhook config yaml + template: + src: apiserver-audit-webhook-config.yaml.j2 + dest: "{{ audit_webhook_config_file }}" + mode: 0640 + when: kubernetes_audit_webhook | default(false) + +# Nginx LB(default), If kubeadm_config_api_fqdn is defined, use other LB by kubeadm controlPlaneEndpoint. +- name: Set kubeadm_config_api_fqdn define + set_fact: + kubeadm_config_api_fqdn: "{{ apiserver_loadbalancer_domain_name | default('lb-apiserver.kubernetes.local') }}" + when: loadbalancer_apiserver is defined + +- name: Set kubeadm api version to v1beta3 + set_fact: + kubeadmConfig_api_version: v1beta3 + +- name: Kubeadm | Create kubeadm config + template: + src: "kubeadm-config.{{ kubeadmConfig_api_version }}.yaml.j2" + dest: "{{ kube_config_dir }}/kubeadm-config.yaml" + mode: 0640 + +- name: Kubeadm | Create directory to store admission control configurations + file: + path: "{{ kube_config_dir }}/admission-controls" + state: directory + mode: 0640 + when: kube_apiserver_admission_control_config_file + +- name: Kubeadm | Push admission control config file + template: + src: "admission-controls.yaml.j2" + dest: "{{ kube_config_dir }}/admission-controls/admission-controls.yaml" + mode: 0640 + when: kube_apiserver_admission_control_config_file + +- name: Kubeadm | Push admission control config files + template: + src: "{{ item | lower }}.yaml.j2" + dest: "{{ kube_config_dir }}/admission-controls/{{ item | lower }}.yaml" + mode: 0640 + when: + - kube_apiserver_admission_control_config_file + - item in kube_apiserver_admission_plugins_needs_configuration + loop: "{{ kube_apiserver_enable_admission_plugins }}" + +- name: Kubeadm | Check apiserver.crt SANs + vars: + apiserver_ips: "{{ apiserver_sans | map('ipaddr') | reject('equalto', False) | list }}" + apiserver_hosts: "{{ apiserver_sans | difference(apiserver_ips) }}" + when: + - kubeadm_already_run.stat.exists + - not kube_external_ca_mode + block: + - name: Kubeadm | Check apiserver.crt SAN IPs + command: + cmd: "openssl x509 -noout -in {{ kube_cert_dir }}/apiserver.crt -checkip {{ item }}" + loop: "{{ apiserver_ips }}" + register: apiserver_sans_ip_check + changed_when: apiserver_sans_ip_check.stdout is not search('does match certificate') + - name: Kubeadm | Check apiserver.crt SAN hosts + command: + cmd: "openssl x509 -noout -in {{ kube_cert_dir }}/apiserver.crt -checkhost {{ item }}" + loop: "{{ apiserver_hosts }}" + register: apiserver_sans_host_check + changed_when: apiserver_sans_host_check.stdout is not search('does match certificate') + +- name: Kubeadm | regenerate apiserver cert 1/2 + file: + state: absent + path: "{{ kube_cert_dir }}/{{ item }}" + with_items: + - apiserver.crt + - apiserver.key + when: + - kubeadm_already_run.stat.exists + - apiserver_sans_ip_check.changed or apiserver_sans_host_check.changed + - not kube_external_ca_mode + +- name: Kubeadm | regenerate apiserver cert 2/2 + command: >- + {{ bin_dir }}/kubeadm + init phase certs apiserver + --config={{ kube_config_dir }}/kubeadm-config.yaml + when: + - kubeadm_already_run.stat.exists + - apiserver_sans_ip_check.changed or apiserver_sans_host_check.changed + - not kube_external_ca_mode + +- name: Kubeadm | Create directory to store kubeadm patches + file: + path: "{{ kubeadm_patches.dest_dir }}" + state: directory + mode: 0640 + when: kubeadm_patches is defined and kubeadm_patches.enabled + +- name: Kubeadm | Copy kubeadm patches from inventory files + copy: + src: "{{ kubeadm_patches.source_dir }}/" + dest: "{{ kubeadm_patches.dest_dir }}" + owner: "root" + mode: 0644 + when: kubeadm_patches is defined and kubeadm_patches.enabled + +- name: Kubeadm | Initialize first master + command: >- + timeout -k {{ kubeadm_init_timeout }} {{ kubeadm_init_timeout }} + {{ bin_dir }}/kubeadm init + --config={{ kube_config_dir }}/kubeadm-config.yaml + --ignore-preflight-errors=all + --skip-phases={{ kubeadm_init_phases_skip | join(',') }} + {{ kube_external_ca_mode | ternary('', '--upload-certs') }} + register: kubeadm_init + # Retry is because upload config sometimes fails + retries: 3 + until: kubeadm_init is succeeded or "field is immutable" in kubeadm_init.stderr + when: inventory_hostname == first_kube_control_plane and not kubeadm_already_run.stat.exists + failed_when: kubeadm_init.rc != 0 and "field is immutable" not in kubeadm_init.stderr + environment: + PATH: "{{ bin_dir }}:{{ ansible_env.PATH }}" + notify: Master | restart kubelet + +- name: Set kubeadm certificate key + set_fact: + kubeadm_certificate_key: "{{ item | regex_search('--certificate-key ([^ ]+)', '\\1') | first }}" + with_items: "{{ hostvars[groups['kube_control_plane'][0]]['kubeadm_init'].stdout_lines | default([]) }}" + when: + - kubeadm_certificate_key is not defined + - (item | trim) is match('.*--certificate-key.*') + +- name: Create hardcoded kubeadm token for joining nodes with 24h expiration (if defined) + shell: >- + {{ bin_dir }}/kubeadm --kubeconfig {{ kube_config_dir }}/admin.conf token delete {{ kubeadm_token }} || :; + {{ bin_dir }}/kubeadm --kubeconfig {{ kube_config_dir }}/admin.conf token create {{ kubeadm_token }} + changed_when: false + when: + - inventory_hostname == first_kube_control_plane + - kubeadm_token is defined + - kubeadm_refresh_token + tags: + - kubeadm_token + +- name: Create kubeadm token for joining nodes with 24h expiration (default) + command: "{{ bin_dir }}/kubeadm --kubeconfig {{ kube_config_dir }}/admin.conf token create" + changed_when: false + register: temp_token + retries: 5 + delay: 5 + until: temp_token is succeeded + delegate_to: "{{ first_kube_control_plane }}" + when: kubeadm_token is not defined + tags: + - kubeadm_token + +- name: Set kubeadm_token + set_fact: + kubeadm_token: "{{ temp_token.stdout }}" + when: temp_token.stdout is defined + tags: + - kubeadm_token + +- name: PodSecurityPolicy | install PodSecurityPolicy + include_tasks: psp-install.yml + when: + - podsecuritypolicy_enabled + - inventory_hostname == first_kube_control_plane + +- name: Kubeadm | Join other masters + include_tasks: kubeadm-secondary.yml + +- name: Kubeadm | upgrade kubernetes cluster + include_tasks: kubeadm-upgrade.yml + when: + - upgrade_cluster_setup + - kubeadm_already_run.stat.exists + +# FIXME(mattymo): from docs: If you don't want to taint your control-plane node, set this field to an empty slice, i.e. `taints: {}` in the YAML file. +- name: Kubeadm | Remove taint for master with node role + command: "{{ kubectl }} taint node {{ inventory_hostname }} {{ item }}" + delegate_to: "{{ first_kube_control_plane }}" + with_items: + - "node-role.kubernetes.io/master:NoSchedule-" + - "node-role.kubernetes.io/control-plane:NoSchedule-" + when: inventory_hostname in groups['kube_node'] + failed_when: false diff --git a/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-upgrade.yml b/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-upgrade.yml new file mode 100644 index 0000000..12ab0b9 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/tasks/kubeadm-upgrade.yml @@ -0,0 +1,78 @@ +--- +- name: Kubeadm | Check api is up + uri: + url: "https://{{ ip | default(fallback_ips[inventory_hostname]) }}:{{ kube_apiserver_port }}/healthz" + validate_certs: false + when: inventory_hostname in groups['kube_control_plane'] + register: _result + retries: 60 + delay: 5 + until: _result.status == 200 + +- name: Kubeadm | Upgrade first master + command: >- + timeout -k 600s 600s + {{ bin_dir }}/kubeadm + upgrade apply -y {{ kube_version }} + --certificate-renewal={{ kubeadm_upgrade_auto_cert_renewal }} + --config={{ kube_config_dir }}/kubeadm-config.yaml + --ignore-preflight-errors=all + --allow-experimental-upgrades + --etcd-upgrade={{ (etcd_deployment_type == "kubeadm") | bool | lower }} + {% if kubeadm_patches is defined and kubeadm_patches.enabled %}--patches={{ kubeadm_patches.dest_dir }}{% endif %} + --force + register: kubeadm_upgrade + # Retry is because upload config sometimes fails + retries: 3 + until: kubeadm_upgrade.rc == 0 + when: inventory_hostname == first_kube_control_plane + failed_when: kubeadm_upgrade.rc != 0 and "field is immutable" not in kubeadm_upgrade.stderr + environment: + PATH: "{{ bin_dir }}:{{ ansible_env.PATH }}" + notify: Master | restart kubelet + +- name: Kubeadm | Upgrade other masters + command: >- + timeout -k 600s 600s + {{ bin_dir }}/kubeadm + upgrade apply -y {{ kube_version }} + --certificate-renewal={{ kubeadm_upgrade_auto_cert_renewal }} + --config={{ kube_config_dir }}/kubeadm-config.yaml + --ignore-preflight-errors=all + --allow-experimental-upgrades + --etcd-upgrade={{ (etcd_deployment_type == "kubeadm") | bool | lower }} + {% if kubeadm_patches is defined and kubeadm_patches.enabled %}--patches={{ kubeadm_patches.dest_dir }}{% endif %} + --force + register: kubeadm_upgrade + # Retry is because upload config sometimes fails + retries: 3 + until: kubeadm_upgrade.rc == 0 + when: inventory_hostname != first_kube_control_plane + failed_when: kubeadm_upgrade.rc != 0 and "field is immutable" not in kubeadm_upgrade.stderr + environment: + PATH: "{{ bin_dir }}:{{ ansible_env.PATH }}" + notify: Master | restart kubelet + +- name: Kubeadm | clean kubectl cache to refresh api types + file: + path: "{{ item }}" + state: absent + with_items: + - /root/.kube/cache + - /root/.kube/http-cache + +# FIXME: https://github.com/kubernetes/kubeadm/issues/1318 +- name: Kubeadm | scale down coredns replicas to 0 if not using coredns dns_mode + command: >- + {{ kubectl }} + -n kube-system + scale deployment/coredns --replicas 0 + register: scale_down_coredns + retries: 6 + delay: 5 + until: scale_down_coredns is succeeded + run_once: true + when: + - kubeadm_scale_down_coredns_enabled + - dns_mode not in ['coredns', 'coredns_dual'] + changed_when: false diff --git a/kubespray/roles/kubernetes/control-plane/tasks/kubelet-fix-client-cert-rotation.yml b/kubespray/roles/kubernetes/control-plane/tasks/kubelet-fix-client-cert-rotation.yml new file mode 100644 index 0000000..7d0c1a0 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/tasks/kubelet-fix-client-cert-rotation.yml @@ -0,0 +1,18 @@ +--- +- name: Fixup kubelet client cert rotation 1/2 + lineinfile: + path: "{{ kube_config_dir }}/kubelet.conf" + regexp: '^ client-certificate-data: ' + line: ' client-certificate: /var/lib/kubelet/pki/kubelet-client-current.pem' + backup: yes + notify: + - "Master | reload kubelet" + +- name: Fixup kubelet client cert rotation 2/2 + lineinfile: + path: "{{ kube_config_dir }}/kubelet.conf" + regexp: '^ client-key-data: ' + line: ' client-key: /var/lib/kubelet/pki/kubelet-client-current.pem' + backup: yes + notify: + - "Master | reload kubelet" diff --git a/kubespray/roles/kubernetes/control-plane/tasks/main.yml b/kubespray/roles/kubernetes/control-plane/tasks/main.yml new file mode 100644 index 0000000..4f251a8 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/tasks/main.yml @@ -0,0 +1,108 @@ +--- +- name: Pre-upgrade control plane + import_tasks: pre-upgrade.yml + tags: + - k8s-pre-upgrade + +- name: Create webhook token auth config + template: + src: webhook-token-auth-config.yaml.j2 + dest: "{{ kube_config_dir }}/webhook-token-auth-config.yaml" + mode: 0640 + when: kube_webhook_token_auth | default(false) + +- name: Create webhook authorization config + template: + src: webhook-authorization-config.yaml.j2 + dest: "{{ kube_config_dir }}/webhook-authorization-config.yaml" + mode: 0640 + when: kube_webhook_authorization | default(false) + +- name: Create kube-scheduler config + template: + src: kubescheduler-config.yaml.j2 + dest: "{{ kube_config_dir }}/kubescheduler-config.yaml" + mode: 0644 + +- name: Apply Kubernetes encrypt at rest config + import_tasks: encrypt-at-rest.yml + when: + - kube_encrypt_secret_data + tags: + - kube-apiserver + +- name: Install | Copy kubectl binary from download dir + copy: + src: "{{ downloads.kubectl.dest }}" + dest: "{{ bin_dir }}/kubectl" + mode: 0755 + remote_src: true + tags: + - kubectl + - upgrade + +- name: Install kubectl bash completion + shell: "{{ bin_dir }}/kubectl completion bash >/etc/bash_completion.d/kubectl.sh" + when: ansible_os_family in ["Debian","RedHat"] + tags: + - kubectl + ignore_errors: true # noqa ignore-errors + +- name: Set kubectl bash completion file permissions + file: + path: /etc/bash_completion.d/kubectl.sh + owner: root + group: root + mode: 0755 + when: ansible_os_family in ["Debian","RedHat"] + tags: + - kubectl + - upgrade + ignore_errors: true # noqa ignore-errors + +- name: Disable SecurityContextDeny admission-controller and enable PodSecurityPolicy + set_fact: + kube_apiserver_enable_admission_plugins: "{{ kube_apiserver_enable_admission_plugins | difference(['SecurityContextDeny']) | union(['PodSecurityPolicy']) | unique }}" + when: podsecuritypolicy_enabled + +- name: Define nodes already joined to existing cluster and first_kube_control_plane + import_tasks: define-first-kube-control.yml + +- name: Include kubeadm setup + import_tasks: kubeadm-setup.yml + +- name: Include kubeadm etcd extra tasks + include_tasks: kubeadm-etcd.yml + when: etcd_deployment_type == "kubeadm" + +- name: Include kubeadm secondary server apiserver fixes + include_tasks: kubeadm-fix-apiserver.yml + +- name: Include kubelet client cert rotation fixes + include_tasks: kubelet-fix-client-cert-rotation.yml + when: kubelet_rotate_certificates + +- name: Install script to renew K8S control plane certificates + template: + src: k8s-certs-renew.sh.j2 + dest: "{{ bin_dir }}/k8s-certs-renew.sh" + mode: 0755 + +- name: Renew K8S control plane certificates monthly 1/2 + template: + src: "{{ item }}.j2" + dest: "/etc/systemd/system/{{ item }}" + mode: 0644 + with_items: + - k8s-certs-renew.service + - k8s-certs-renew.timer + register: k8s_certs_units + when: auto_renew_certificates + +- name: Renew K8S control plane certificates monthly 2/2 + systemd: + name: k8s-certs-renew.timer + enabled: yes + state: started + daemon_reload: "{{ k8s_certs_units is changed }}" + when: auto_renew_certificates diff --git a/kubespray/roles/kubernetes/control-plane/tasks/pre-upgrade.yml b/kubespray/roles/kubernetes/control-plane/tasks/pre-upgrade.yml new file mode 100644 index 0000000..2d7dce5 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/tasks/pre-upgrade.yml @@ -0,0 +1,21 @@ +--- +- name: "Pre-upgrade | Delete master manifests if etcd secrets changed" + file: + path: "/etc/kubernetes/manifests/{{ item }}.manifest" + state: absent + with_items: + - ["kube-apiserver", "kube-controller-manager", "kube-scheduler"] + register: kube_apiserver_manifest_replaced + when: etcd_secret_changed | default(false) + +- name: "Pre-upgrade | Delete master containers forcefully" # noqa no-handler + shell: "set -o pipefail && docker ps -af name=k8s_{{ item }}* -q | xargs --no-run-if-empty docker rm -f" + args: + executable: /bin/bash + with_items: + - ["kube-apiserver", "kube-controller-manager", "kube-scheduler"] + when: kube_apiserver_manifest_replaced.changed + register: remove_master_container + retries: 10 + until: remove_master_container.rc == 0 + delay: 1 diff --git a/kubespray/roles/kubernetes/control-plane/tasks/psp-install.yml b/kubespray/roles/kubernetes/control-plane/tasks/psp-install.yml new file mode 100644 index 0000000..4a990f8 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/tasks/psp-install.yml @@ -0,0 +1,38 @@ +--- +- name: Check AppArmor status + command: which apparmor_parser + register: apparmor_status + failed_when: false + changed_when: apparmor_status.rc != 0 + +- name: Set apparmor_enabled + set_fact: + apparmor_enabled: "{{ apparmor_status.rc == 0 }}" + +- name: Render templates for PodSecurityPolicy + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0640 + register: psp_manifests + with_items: + - {file: psp.yml, type: psp, name: psp} + - {file: psp-cr.yml, type: clusterrole, name: psp-cr} + - {file: psp-crb.yml, type: rolebinding, name: psp-crb} + +- name: Add policies, roles, bindings for PodSecurityPolicy + kube: + name: "{{ item.item.name }}" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + register: result + until: result is succeeded + retries: 10 + delay: 6 + with_items: "{{ psp_manifests.results }}" + environment: + KUBECONFIG: "{{ kube_config_dir }}/admin.conf" + loop_control: + label: "{{ item.item.file }}" diff --git a/kubespray/roles/kubernetes/control-plane/templates/admission-controls.yaml.j2 b/kubespray/roles/kubernetes/control-plane/templates/admission-controls.yaml.j2 new file mode 100644 index 0000000..fc4d0ef --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/admission-controls.yaml.j2 @@ -0,0 +1,9 @@ +apiVersion: apiserver.config.k8s.io/v1 +kind: AdmissionConfiguration +plugins: +{% for plugin in kube_apiserver_enable_admission_plugins %} +{% if plugin in kube_apiserver_admission_plugins_needs_configuration %} +- name: {{ plugin }} + path: {{ kube_config_dir }}/{{ plugin | lower }}.yaml +{% endif %} +{% endfor %} diff --git a/kubespray/roles/kubernetes/control-plane/templates/apiserver-audit-policy.yaml.j2 b/kubespray/roles/kubernetes/control-plane/templates/apiserver-audit-policy.yaml.j2 new file mode 100644 index 0000000..ca7bcf8 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/apiserver-audit-policy.yaml.j2 @@ -0,0 +1,129 @@ +apiVersion: audit.k8s.io/v1 +kind: Policy +rules: +{% if audit_policy_custom_rules is defined and audit_policy_custom_rules != "" %} +{{ audit_policy_custom_rules | indent(2, true) }} +{% else %} + # The following requests were manually identified as high-volume and low-risk, + # so drop them. + - level: None + users: ["system:kube-proxy"] + verbs: ["watch"] + resources: + - group: "" # core + resources: ["endpoints", "services", "services/status"] + - level: None + # Ingress controller reads `configmaps/ingress-uid` through the unsecured port. + # TODO(#46983): Change this to the ingress controller service account. + users: ["system:unsecured"] + namespaces: ["kube-system"] + verbs: ["get"] + resources: + - group: "" # core + resources: ["configmaps"] + - level: None + users: ["kubelet"] # legacy kubelet identity + verbs: ["get"] + resources: + - group: "" # core + resources: ["nodes", "nodes/status"] + - level: None + userGroups: ["system:nodes"] + verbs: ["get"] + resources: + - group: "" # core + resources: ["nodes", "nodes/status"] + - level: None + users: + - system:kube-controller-manager + - system:kube-scheduler + - system:serviceaccount:kube-system:endpoint-controller + verbs: ["get", "update"] + namespaces: ["kube-system"] + resources: + - group: "" # core + resources: ["endpoints"] + - level: None + users: ["system:apiserver"] + verbs: ["get"] + resources: + - group: "" # core + resources: ["namespaces", "namespaces/status", "namespaces/finalize"] + # Don't log HPA fetching metrics. + - level: None + users: + - system:kube-controller-manager + verbs: ["get", "list"] + resources: + - group: "metrics.k8s.io" + # Don't log these read-only URLs. + - level: None + nonResourceURLs: + - /healthz* + - /version + - /swagger* + # Don't log events requests. + - level: None + resources: + - group: "" # core + resources: ["events"] + # Secrets, ConfigMaps, TokenRequest and TokenReviews can contain sensitive & binary data, + # so only log at the Metadata level. + - level: Metadata + resources: + - group: "" # core + resources: ["secrets", "configmaps", "serviceaccounts/token"] + - group: authentication.k8s.io + resources: ["tokenreviews"] + omitStages: + - "RequestReceived" + # Get responses can be large; skip them. + - level: Request + verbs: ["get", "list", "watch"] + resources: + - group: "" # core + - group: "admissionregistration.k8s.io" + - group: "apiextensions.k8s.io" + - group: "apiregistration.k8s.io" + - group: "apps" + - group: "authentication.k8s.io" + - group: "authorization.k8s.io" + - group: "autoscaling" + - group: "batch" + - group: "certificates.k8s.io" + - group: "extensions" + - group: "metrics.k8s.io" + - group: "networking.k8s.io" + - group: "policy" + - group: "rbac.authorization.k8s.io" + - group: "settings.k8s.io" + - group: "storage.k8s.io" + omitStages: + - "RequestReceived" + # Default level for known APIs + - level: RequestResponse + resources: + - group: "" # core + - group: "admissionregistration.k8s.io" + - group: "apiextensions.k8s.io" + - group: "apiregistration.k8s.io" + - group: "apps" + - group: "authentication.k8s.io" + - group: "authorization.k8s.io" + - group: "autoscaling" + - group: "batch" + - group: "certificates.k8s.io" + - group: "extensions" + - group: "metrics.k8s.io" + - group: "networking.k8s.io" + - group: "policy" + - group: "rbac.authorization.k8s.io" + - group: "settings.k8s.io" + - group: "storage.k8s.io" + omitStages: + - "RequestReceived" + # Default level for all other requests. + - level: Metadata + omitStages: + - "RequestReceived" +{% endif %} diff --git a/kubespray/roles/kubernetes/control-plane/templates/apiserver-audit-webhook-config.yaml.j2 b/kubespray/roles/kubernetes/control-plane/templates/apiserver-audit-webhook-config.yaml.j2 new file mode 100644 index 0000000..cd8208e --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/apiserver-audit-webhook-config.yaml.j2 @@ -0,0 +1,17 @@ +apiVersion: v1 +kind: Config +clusters: +- cluster: + server: {{ audit_webhook_server_url }} +{% for key in audit_webhook_server_extra_args %} + {{ key }}: "{{ audit_webhook_server_extra_args[key] }}" +{% endfor %} + name: auditsink +contexts: +- context: + cluster: auditsink + user: "" + name: default-context +current-context: default-context +preferences: {} +users: [] diff --git a/kubespray/roles/kubernetes/control-plane/templates/eventratelimit.yaml.j2 b/kubespray/roles/kubernetes/control-plane/templates/eventratelimit.yaml.j2 new file mode 100644 index 0000000..0d78670 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/eventratelimit.yaml.j2 @@ -0,0 +1,11 @@ +apiVersion: eventratelimit.admission.k8s.io/v1alpha1 +kind: Configuration +limits: +{% for limit in kube_apiserver_admission_event_rate_limits.values() %} +- type: {{ limit.type }} + qps: {{ limit.qps }} + burst: {{ limit.burst }} +{% if limit.cache_size is defined %} + cacheSize: {{ limit.cache_size }} +{% endif %} +{% endfor %} diff --git a/kubespray/roles/kubernetes/control-plane/templates/k8s-certs-renew.service.j2 b/kubespray/roles/kubernetes/control-plane/templates/k8s-certs-renew.service.j2 new file mode 100644 index 0000000..64610c2 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/k8s-certs-renew.service.j2 @@ -0,0 +1,6 @@ +[Unit] +Description=Renew K8S control plane certificates + +[Service] +Type=oneshot +ExecStart={{ bin_dir }}/k8s-certs-renew.sh diff --git a/kubespray/roles/kubernetes/control-plane/templates/k8s-certs-renew.sh.j2 b/kubespray/roles/kubernetes/control-plane/templates/k8s-certs-renew.sh.j2 new file mode 100644 index 0000000..b2c7c77 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/k8s-certs-renew.sh.j2 @@ -0,0 +1,23 @@ +#!/bin/bash + +echo "## Expiration before renewal ##" +{{ bin_dir }}/kubeadm certs check-expiration + +echo "## Renewing certificates managed by kubeadm ##" +{{ bin_dir }}/kubeadm certs renew all + +echo "## Restarting control plane pods managed by kubeadm ##" +{% if container_manager == "docker" %} +{{ docker_bin_dir }}/docker ps -af 'name=k8s_POD_(kube-apiserver|kube-controller-manager|kube-scheduler|etcd)-*' -q | /usr/bin/xargs {{ docker_bin_dir }}/docker rm -f +{% else %} +{{ bin_dir }}/crictl pods --namespace kube-system --name 'kube-scheduler-*|kube-controller-manager-*|kube-apiserver-*|etcd-*' -q | /usr/bin/xargs {{ bin_dir }}/crictl rmp -f +{% endif %} + +echo "## Updating /root/.kube/config ##" +cp {{ kube_config_dir }}/admin.conf /root/.kube/config + +echo "## Waiting for apiserver to be up again ##" +until printf "" 2>>/dev/null >>/dev/tcp/127.0.0.1/{{ kube_apiserver_port | default(6443) }}; do sleep 1; done + +echo "## Expiration after renewal ##" +{{ bin_dir }}/kubeadm certs check-expiration diff --git a/kubespray/roles/kubernetes/control-plane/templates/k8s-certs-renew.timer.j2 b/kubespray/roles/kubernetes/control-plane/templates/k8s-certs-renew.timer.j2 new file mode 100644 index 0000000..904f007 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/k8s-certs-renew.timer.j2 @@ -0,0 +1,8 @@ +[Unit] +Description=Timer to renew K8S control plane certificates + +[Timer] +OnCalendar={{ auto_renew_certificates_systemd_calendar }} + +[Install] +WantedBy=multi-user.target diff --git a/kubespray/roles/kubernetes/control-plane/templates/kubeadm-config.v1beta3.yaml.j2 b/kubespray/roles/kubernetes/control-plane/templates/kubeadm-config.v1beta3.yaml.j2 new file mode 100644 index 0000000..6410571 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/kubeadm-config.v1beta3.yaml.j2 @@ -0,0 +1,456 @@ +apiVersion: kubeadm.k8s.io/v1beta3 +kind: InitConfiguration +{% if kubeadm_token is defined %} +bootstrapTokens: +- token: "{{ kubeadm_token }}" + description: "kubespray kubeadm bootstrap token" + ttl: "24h" +{% endif %} +localAPIEndpoint: + advertiseAddress: {{ kube_apiserver_address }} + bindPort: {{ kube_apiserver_port }} +{% if kubeadm_certificate_key is defined %} +certificateKey: {{ kubeadm_certificate_key }} +{% endif %} +nodeRegistration: +{% if kube_override_hostname | default('') %} + name: {{ kube_override_hostname }} +{% endif %} +{% if inventory_hostname in groups['kube_control_plane'] and inventory_hostname not in groups['kube_node'] %} + taints: + - effect: NoSchedule + key: node-role.kubernetes.io/master + - effect: NoSchedule + key: node-role.kubernetes.io/control-plane +{% else %} + taints: [] +{% endif %} + criSocket: {{ cri_socket }} +{% if cloud_provider is defined and cloud_provider in ["external"] %} + kubeletExtraArgs: + cloud-provider: external +{% endif %} +{% if kubeadm_patches is defined and kubeadm_patches.enabled %} +patches: + directory: {{ kubeadm_patches.dest_dir }} +{% endif %} +--- +apiVersion: kubeadm.k8s.io/v1beta3 +kind: ClusterConfiguration +clusterName: {{ cluster_name }} +etcd: +{% if etcd_deployment_type != "kubeadm" %} + external: + endpoints: +{% for endpoint in etcd_access_addresses.split(',') %} + - {{ endpoint }} +{% endfor %} + caFile: {{ etcd_cert_dir }}/{{ kube_etcd_cacert_file }} + certFile: {{ etcd_cert_dir }}/{{ kube_etcd_cert_file }} + keyFile: {{ etcd_cert_dir }}/{{ kube_etcd_key_file }} +{% elif etcd_deployment_type == "kubeadm" %} + local: + imageRepository: "{{ etcd_image_repo | regex_replace("/etcd$","") }}" + imageTag: "{{ etcd_image_tag }}" + dataDir: "{{ etcd_data_dir }}" + extraArgs: + metrics: {{ etcd_metrics }} + election-timeout: "{{ etcd_election_timeout }}" + heartbeat-interval: "{{ etcd_heartbeat_interval }}" + auto-compaction-retention: "{{ etcd_compaction_retention }}" +{% if etcd_listen_metrics_urls is defined %} + listen-metrics-urls: "{{ etcd_listen_metrics_urls }}" +{% endif %} +{% if etcd_snapshot_count is defined %} + snapshot-count: "{{ etcd_snapshot_count }}" +{% endif %} +{% if etcd_quota_backend_bytes is defined %} + quota-backend-bytes: "{{ etcd_quota_backend_bytes }}" +{% endif %} +{% if etcd_max_request_bytes is defined %} + max-request-bytes: "{{ etcd_max_request_bytes }}" +{% endif %} +{% if etcd_log_level is defined %} + log-level: "{{ etcd_log_level }}" +{% endif %} +{% for key, value in etcd_extra_vars.items() %} + {{ key }}: "{{ value }}" +{% endfor %} + serverCertSANs: +{% for san in etcd_cert_alt_names %} + - {{ san }} +{% endfor %} +{% for san in etcd_cert_alt_ips %} + - {{ san }} +{% endfor %} + peerCertSANs: +{% for san in etcd_cert_alt_names %} + - {{ san }} +{% endfor %} +{% for san in etcd_cert_alt_ips %} + - {{ san }} +{% endfor %} +{% endif %} +dns: + imageRepository: {{ coredns_image_repo | regex_replace('/coredns(?!/coredns).*$', '') }} + imageTag: {{ coredns_image_tag }} +networking: + dnsDomain: {{ dns_domain }} + serviceSubnet: "{{ kube_service_addresses }}{{ ',' + kube_service_addresses_ipv6 if enable_dual_stack_networks else '' }}" +{% if kube_network_plugin is defined and kube_network_plugin not in ["kube-ovn"] %} + podSubnet: "{{ kube_pods_subnet }}{{ ',' + kube_pods_subnet_ipv6 if enable_dual_stack_networks else '' }}" +{% endif %} +{% if kubeadm_feature_gates %} +featureGates: +{% for feature in kubeadm_feature_gates %} + {{ feature | replace("=", ": ") }} +{% endfor %} +{% endif %} +kubernetesVersion: {{ kube_version }} +{% if kubeadm_config_api_fqdn is defined %} +controlPlaneEndpoint: {{ kubeadm_config_api_fqdn }}:{{ loadbalancer_apiserver.port | default(kube_apiserver_port) }} +{% else %} +controlPlaneEndpoint: {{ ip | default(fallback_ips[inventory_hostname]) }}:{{ kube_apiserver_port }} +{% endif %} +certificatesDir: {{ kube_cert_dir }} +imageRepository: {{ kube_image_repo }} +apiServer: + extraArgs: +{% if kube_apiserver_pod_eviction_not_ready_timeout_seconds is defined %} + default-not-ready-toleration-seconds: "{{ kube_apiserver_pod_eviction_not_ready_timeout_seconds }}" +{% endif %} +{% if kube_apiserver_pod_eviction_unreachable_timeout_seconds is defined %} + default-unreachable-toleration-seconds: "{{ kube_apiserver_pod_eviction_unreachable_timeout_seconds }}" +{% endif %} +{% if kube_api_anonymous_auth is defined %} + anonymous-auth: "{{ kube_api_anonymous_auth }}" +{% endif %} + authorization-mode: {{ authorization_modes | join(',') }} + bind-address: {{ kube_apiserver_bind_address }} +{% if kube_apiserver_enable_admission_plugins | length > 0 %} + enable-admission-plugins: {{ kube_apiserver_enable_admission_plugins | join(',') }} +{% endif %} +{% if kube_apiserver_admission_control_config_file %} + admission-control-config-file: {{ kube_config_dir }}/admission-controls.yaml +{% endif %} +{% if kube_apiserver_disable_admission_plugins | length > 0 %} + disable-admission-plugins: {{ kube_apiserver_disable_admission_plugins | join(',') }} +{% endif %} + apiserver-count: "{{ kube_apiserver_count }}" + endpoint-reconciler-type: lease +{% if etcd_events_cluster_enabled %} + etcd-servers-overrides: "/events#{{ etcd_events_access_addresses_semicolon }}" +{% endif %} + service-node-port-range: {{ kube_apiserver_node_port_range }} + service-cluster-ip-range: "{{ kube_service_addresses }}{{ ',' + kube_service_addresses_ipv6 if enable_dual_stack_networks else '' }}" + kubelet-preferred-address-types: "{{ kubelet_preferred_address_types }}" + profiling: "{{ kube_profiling }}" + request-timeout: "{{ kube_apiserver_request_timeout }}" + enable-aggregator-routing: "{{ kube_api_aggregator_routing }}" +{% if kube_token_auth | default(true) %} + token-auth-file: {{ kube_token_dir }}/known_tokens.csv +{% endif %} +{% if kube_apiserver_service_account_lookup %} + service-account-lookup: "{{ kube_apiserver_service_account_lookup }}" +{% endif %} +{% if kube_oidc_auth | default(false) and kube_oidc_url is defined and kube_oidc_client_id is defined %} + oidc-issuer-url: "{{ kube_oidc_url }}" + oidc-client-id: "{{ kube_oidc_client_id }}" +{% if kube_oidc_ca_file is defined %} + oidc-ca-file: "{{ kube_oidc_ca_file }}" +{% endif %} +{% if kube_oidc_username_claim is defined %} + oidc-username-claim: "{{ kube_oidc_username_claim }}" +{% endif %} +{% if kube_oidc_groups_claim is defined %} + oidc-groups-claim: "{{ kube_oidc_groups_claim }}" +{% endif %} +{% if kube_oidc_username_prefix is defined %} + oidc-username-prefix: "{{ kube_oidc_username_prefix }}" +{% endif %} +{% if kube_oidc_groups_prefix is defined %} + oidc-groups-prefix: "{{ kube_oidc_groups_prefix }}" +{% endif %} +{% endif %} +{% if kube_webhook_token_auth | default(false) %} + authentication-token-webhook-config-file: {{ kube_config_dir }}/webhook-token-auth-config.yaml +{% endif %} +{% if kube_webhook_authorization | default(false) %} + authorization-webhook-config-file: {{ kube_config_dir }}/webhook-authorization-config.yaml +{% endif %} +{% if kube_encrypt_secret_data %} + encryption-provider-config: {{ kube_cert_dir }}/secrets_encryption.yaml +{% endif %} + storage-backend: {{ kube_apiserver_storage_backend }} +{% if kube_api_runtime_config | length > 0 %} + runtime-config: {{ kube_api_runtime_config | join(',') }} +{% endif %} + allow-privileged: "true" +{% if kubernetes_audit or kubernetes_audit_webhook %} + audit-policy-file: {{ audit_policy_file }} +{% endif %} +{% if kubernetes_audit %} + audit-log-path: "{{ audit_log_path }}" + audit-log-maxage: "{{ audit_log_maxage }}" + audit-log-maxbackup: "{{ audit_log_maxbackups }}" + audit-log-maxsize: "{{ audit_log_maxsize }}" +{% endif %} +{% if kubernetes_audit_webhook %} + audit-webhook-config-file: {{ audit_webhook_config_file }} + audit-webhook-mode: {{ audit_webhook_mode }} +{% if audit_webhook_mode == "batch" %} + audit-webhook-batch-max-size: "{{ audit_webhook_batch_max_size }}" + audit-webhook-batch-max-wait: "{{ audit_webhook_batch_max_wait }}" +{% endif %} +{% endif %} +{% for key in kube_kubeadm_apiserver_extra_args %} + {{ key }}: "{{ kube_kubeadm_apiserver_extra_args[key] }}" +{% endfor %} +{% if kube_apiserver_feature_gates or kube_feature_gates %} + feature-gates: "{{ kube_apiserver_feature_gates | default(kube_feature_gates, true) | join(',') }}" +{% endif %} +{% if cloud_provider is defined and cloud_provider in ["openstack", "azure", "vsphere", "aws", "gce"] %} + cloud-provider: {{ cloud_provider }} + cloud-config: {{ kube_config_dir }}/cloud_config +{% endif %} +{% if tls_min_version is defined %} + tls-min-version: {{ tls_min_version }} +{% endif %} +{% if tls_cipher_suites is defined %} + tls-cipher-suites: {% for tls in tls_cipher_suites %}{{ tls }}{{ "," if not loop.last else "" }}{% endfor %} + +{% endif %} +{% if event_ttl_duration is defined %} + event-ttl: {{ event_ttl_duration }} +{% endif %} +{% if kubelet_rotate_server_certificates %} + kubelet-certificate-authority: {{ kube_cert_dir }}/ca.crt +{% endif %} +{% if kubernetes_audit or kube_token_auth | default(true) or kube_webhook_token_auth | default(false) or ( cloud_provider is defined and cloud_provider in ["openstack", "azure", "vsphere", "aws", "gce"] ) or apiserver_extra_volumes or ssl_ca_dirs | length %} + extraVolumes: +{% if cloud_provider is defined and cloud_provider in ["openstack", "azure", "vsphere", "aws", "gce"] %} + - name: cloud-config + hostPath: {{ kube_config_dir }}/cloud_config + mountPath: {{ kube_config_dir }}/cloud_config +{% endif %} +{% if kube_token_auth | default(true) %} + - name: token-auth-config + hostPath: {{ kube_token_dir }} + mountPath: {{ kube_token_dir }} +{% endif %} +{% if kube_webhook_token_auth | default(false) %} + - name: webhook-token-auth-config + hostPath: {{ kube_config_dir }}/webhook-token-auth-config.yaml + mountPath: {{ kube_config_dir }}/webhook-token-auth-config.yaml +{% endif %} +{% if kube_webhook_authorization | default(false) %} + - name: webhook-authorization-config + hostPath: {{ kube_config_dir }}/webhook-authorization-config.yaml + mountPath: {{ kube_config_dir }}/webhook-authorization-config.yaml +{% endif %} +{% if kubernetes_audit or kubernetes_audit_webhook %} + - name: {{ audit_policy_name }} + hostPath: {{ audit_policy_hostpath }} + mountPath: {{ audit_policy_mountpath }} +{% if audit_log_path != "-" %} + - name: {{ audit_log_name }} + hostPath: {{ audit_log_hostpath }} + mountPath: {{ audit_log_mountpath }} + readOnly: false +{% endif %} +{% endif %} +{% if kube_apiserver_admission_control_config_file %} + - name: admission-control-configs + hostPath: {{ kube_config_dir }}/admission-controls + mountPath: {{ kube_config_dir }} + readOnly: false + pathType: DirectoryOrCreate +{% endif %} +{% for volume in apiserver_extra_volumes %} + - name: {{ volume.name }} + hostPath: {{ volume.hostPath }} + mountPath: {{ volume.mountPath }} + readOnly: {{ volume.readOnly | d(not (volume.writable | d(false))) }} +{% endfor %} +{% if ssl_ca_dirs | length %} +{% for dir in ssl_ca_dirs %} + - name: {{ dir | regex_replace('^/(.*)$', '\\1' ) | regex_replace('/', '-') }} + hostPath: {{ dir }} + mountPath: {{ dir }} + readOnly: true +{% endfor %} +{% endif %} +{% endif %} + certSANs: +{% for san in apiserver_sans %} + - {{ san }} +{% endfor %} + timeoutForControlPlane: 5m0s +controllerManager: + extraArgs: + node-monitor-grace-period: {{ kube_controller_node_monitor_grace_period }} + node-monitor-period: {{ kube_controller_node_monitor_period }} +{% if kube_network_plugin is defined and kube_network_plugin not in ["kube-ovn"] %} + cluster-cidr: "{{ kube_pods_subnet }}{{ ',' + kube_pods_subnet_ipv6 if enable_dual_stack_networks else '' }}" +{% endif %} + service-cluster-ip-range: "{{ kube_service_addresses }}{{ ',' + kube_service_addresses_ipv6 if enable_dual_stack_networks else '' }}" +{% if enable_dual_stack_networks %} + node-cidr-mask-size-ipv4: "{{ kube_network_node_prefix }}" + node-cidr-mask-size-ipv6: "{{ kube_network_node_prefix_ipv6 }}" +{% else %} + node-cidr-mask-size: "{{ kube_network_node_prefix }}" +{% endif %} + profiling: "{{ kube_profiling }}" + terminated-pod-gc-threshold: "{{ kube_controller_terminated_pod_gc_threshold }}" + bind-address: {{ kube_controller_manager_bind_address }} + leader-elect-lease-duration: {{ kube_controller_manager_leader_elect_lease_duration }} + leader-elect-renew-deadline: {{ kube_controller_manager_leader_elect_renew_deadline }} +{% if kube_controller_feature_gates or kube_feature_gates %} + feature-gates: "{{ kube_controller_feature_gates | default(kube_feature_gates, true) | join(',') }}" +{% endif %} +{% for key in kube_kubeadm_controller_extra_args %} + {{ key }}: "{{ kube_kubeadm_controller_extra_args[key] }}" +{% endfor %} +{% if cloud_provider is defined and cloud_provider in ["openstack", "azure", "vsphere", "aws", "gce"] %} + cloud-provider: {{ cloud_provider }} + cloud-config: {{ kube_config_dir }}/cloud_config +{% endif %} +{% if kube_network_plugin is defined and kube_network_plugin not in ["cloud"] %} + configure-cloud-routes: "false" +{% endif %} +{% if kubelet_flexvolumes_plugins_dir is defined %} + flex-volume-plugin-dir: {{ kubelet_flexvolumes_plugins_dir }} +{% endif %} +{% if tls_min_version is defined %} + tls-min-version: {{ tls_min_version }} +{% endif %} +{% if tls_cipher_suites is defined %} + tls-cipher-suites: {% for tls in tls_cipher_suites %}{{ tls }}{{ "," if not loop.last else "" }}{% endfor %} + +{% endif %} +{% if cloud_provider is defined and cloud_provider in ["openstack", "azure", "vsphere", "aws", "gce"] or controller_manager_extra_volumes %} + extraVolumes: +{% if cloud_provider is defined and cloud_provider in ["openstack"] and openstack_cacert is defined %} + - name: openstackcacert + hostPath: "{{ kube_config_dir }}/openstack-cacert.pem" + mountPath: "{{ kube_config_dir }}/openstack-cacert.pem" +{% endif %} +{% if cloud_provider is defined and cloud_provider in ["openstack", "azure", "vsphere", "aws", "gce"] %} + - name: cloud-config + hostPath: {{ kube_config_dir }}/cloud_config + mountPath: {{ kube_config_dir }}/cloud_config +{% endif %} +{% for volume in controller_manager_extra_volumes %} + - name: {{ volume.name }} + hostPath: {{ volume.hostPath }} + mountPath: {{ volume.mountPath }} + readOnly: {{ volume.readOnly | d(not (volume.writable | d(false))) }} +{% endfor %} +{% endif %} +scheduler: + extraArgs: + bind-address: {{ kube_scheduler_bind_address }} + config: {{ kube_config_dir }}/kubescheduler-config.yaml +{% if kube_scheduler_feature_gates or kube_feature_gates %} + feature-gates: "{{ kube_scheduler_feature_gates | default(kube_feature_gates, true) | join(',') }}" +{% endif %} + profiling: "{{ kube_profiling }}" +{% if kube_kubeadm_scheduler_extra_args | length > 0 %} +{% for key in kube_kubeadm_scheduler_extra_args %} + {{ key }}: "{{ kube_kubeadm_scheduler_extra_args[key] }}" +{% endfor %} +{% endif %} +{% if tls_min_version is defined %} + tls-min-version: {{ tls_min_version }} +{% endif %} +{% if tls_cipher_suites is defined %} + tls-cipher-suites: {% for tls in tls_cipher_suites %}{{ tls }}{{ "," if not loop.last else "" }}{% endfor %} + +{% endif %} + extraVolumes: + - name: kubescheduler-config + hostPath: {{ kube_config_dir }}/kubescheduler-config.yaml + mountPath: {{ kube_config_dir }}/kubescheduler-config.yaml + readOnly: true +{% if scheduler_extra_volumes %} +{% for volume in scheduler_extra_volumes %} + - name: {{ volume.name }} + hostPath: {{ volume.hostPath }} + mountPath: {{ volume.mountPath }} + readOnly: {{ volume.readOnly | d(not (volume.writable | d(false))) }} +{% endfor %} +{% endif %} +--- +apiVersion: kubeproxy.config.k8s.io/v1alpha1 +kind: KubeProxyConfiguration +bindAddress: {{ kube_proxy_bind_address }} +clientConnection: + acceptContentTypes: {{ kube_proxy_client_accept_content_types }} + burst: {{ kube_proxy_client_burst }} + contentType: {{ kube_proxy_client_content_type }} + kubeconfig: {{ kube_proxy_client_kubeconfig }} + qps: {{ kube_proxy_client_qps }} +{% if kube_network_plugin is defined and kube_network_plugin not in ["kube-ovn"] %} +clusterCIDR: "{{ kube_pods_subnet }}{{ ',' + kube_pods_subnet_ipv6 if enable_dual_stack_networks else '' }}" +{% endif %} +configSyncPeriod: {{ kube_proxy_config_sync_period }} +conntrack: + maxPerCore: {{ kube_proxy_conntrack_max_per_core }} + min: {{ kube_proxy_conntrack_min }} + tcpCloseWaitTimeout: {{ kube_proxy_conntrack_tcp_close_wait_timeout }} + tcpEstablishedTimeout: {{ kube_proxy_conntrack_tcp_established_timeout }} +enableProfiling: {{ kube_proxy_enable_profiling }} +healthzBindAddress: {{ kube_proxy_healthz_bind_address }} +hostnameOverride: {{ kube_override_hostname }} +iptables: + masqueradeAll: {{ kube_proxy_masquerade_all }} + masqueradeBit: {{ kube_proxy_masquerade_bit }} + minSyncPeriod: {{ kube_proxy_min_sync_period }} + syncPeriod: {{ kube_proxy_sync_period }} +ipvs: + excludeCIDRs: {{ kube_proxy_exclude_cidrs }} + minSyncPeriod: {{ kube_proxy_min_sync_period }} + scheduler: {{ kube_proxy_scheduler }} + syncPeriod: {{ kube_proxy_sync_period }} + strictARP: {{ kube_proxy_strict_arp }} + tcpTimeout: {{ kube_proxy_tcp_timeout }} + tcpFinTimeout: {{ kube_proxy_tcp_fin_timeout }} + udpTimeout: {{ kube_proxy_udp_timeout }} +metricsBindAddress: {{ kube_proxy_metrics_bind_address }} +mode: {{ kube_proxy_mode }} +nodePortAddresses: {{ kube_proxy_nodeport_addresses }} +oomScoreAdj: {{ kube_proxy_oom_score_adj }} +portRange: {{ kube_proxy_port_range }} +{% if kube_proxy_feature_gates or kube_feature_gates %} +{% set feature_gates = ( kube_proxy_feature_gates | default(kube_feature_gates, true) ) %} +featureGates: +{% for feature in feature_gates %} + {{ feature | replace("=", ": ") }} +{% endfor %} +{% endif %} +{# DNS settings for kubelet #} +{% if enable_nodelocaldns %} +{% set kubelet_cluster_dns = [nodelocaldns_ip] %} +{% elif dns_mode in ['coredns'] %} +{% set kubelet_cluster_dns = [skydns_server] %} +{% elif dns_mode == 'coredns_dual' %} +{% set kubelet_cluster_dns = [skydns_server,skydns_server_secondary] %} +{% elif dns_mode == 'manual' %} +{% set kubelet_cluster_dns = [manual_dns_server] %} +{% else %} +{% set kubelet_cluster_dns = [] %} +{% endif %} +--- +apiVersion: kubelet.config.k8s.io/v1beta1 +kind: KubeletConfiguration +clusterDNS: +{% for dns_address in kubelet_cluster_dns %} +- {{ dns_address }} +{% endfor %} +{% if kubelet_feature_gates or kube_feature_gates %} +{% set feature_gates = ( kubelet_feature_gates | default(kube_feature_gates, true) ) %} +featureGates: +{% for feature in feature_gates %} + {{ feature | replace("=", ": ") }} +{% endfor %} +{% endif %} diff --git a/kubespray/roles/kubernetes/control-plane/templates/kubeadm-controlplane.v1beta3.yaml.j2 b/kubespray/roles/kubernetes/control-plane/templates/kubeadm-controlplane.v1beta3.yaml.j2 new file mode 100644 index 0000000..fc696ae --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/kubeadm-controlplane.v1beta3.yaml.j2 @@ -0,0 +1,34 @@ +apiVersion: kubeadm.k8s.io/v1beta3 +kind: JoinConfiguration +discovery: + bootstrapToken: +{% if kubeadm_config_api_fqdn is defined %} + apiServerEndpoint: {{ kubeadm_config_api_fqdn }}:{{ loadbalancer_apiserver.port | default(kube_apiserver_port) }} +{% else %} + apiServerEndpoint: {{ kubeadm_discovery_address }} +{% endif %} + token: {{ kubeadm_token }} + unsafeSkipCAVerification: true + timeout: {{ discovery_timeout }} + tlsBootstrapToken: {{ kubeadm_token }} +controlPlane: + localAPIEndpoint: + advertiseAddress: {{ kube_apiserver_address }} + bindPort: {{ kube_apiserver_port }} + certificateKey: {{ kubeadm_certificate_key }} +nodeRegistration: + name: {{ kube_override_hostname | default(inventory_hostname) }} + criSocket: {{ cri_socket }} +{% if inventory_hostname in groups['kube_control_plane'] and inventory_hostname not in groups['kube_node'] %} + taints: + - effect: NoSchedule + key: node-role.kubernetes.io/master + - effect: NoSchedule + key: node-role.kubernetes.io/control-plane +{% else %} + taints: [] +{% endif %} +{% if kubeadm_patches is defined and kubeadm_patches.enabled %} +patches: + directory: {{ kubeadm_patches.dest_dir }} +{% endif %} diff --git a/kubespray/roles/kubernetes/control-plane/templates/kubescheduler-config.yaml.j2 b/kubespray/roles/kubernetes/control-plane/templates/kubescheduler-config.yaml.j2 new file mode 100644 index 0000000..a517fe8 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/kubescheduler-config.yaml.j2 @@ -0,0 +1,24 @@ +apiVersion: kubescheduler.config.k8s.io/v1 +kind: KubeSchedulerConfiguration +clientConnection: + kubeconfig: "{{ kube_config_dir }}/scheduler.conf" +{% for key in kube_scheduler_client_conn_extra_opts %} + {{ key }}: {{ kube_scheduler_client_conn_extra_opts[key] }} +{% endfor %} +{% if kube_scheduler_extenders %} +extenders: +{{ kube_scheduler_extenders | to_nice_yaml(indent=2, width=256) }} +{% endif %} +leaderElection: + leaseDuration: {{ kube_scheduler_leader_elect_lease_duration }} + renewDeadline: {{ kube_scheduler_leader_elect_renew_deadline }} +{% for key in kube_scheduler_leader_elect_extra_opts %} + {{ key }}: {{ kube_scheduler_leader_elect_extra_opts[key] }} +{% endfor %} +{% if kube_scheduler_profiles %} +profiles: +{{ kube_scheduler_profiles | to_nice_yaml(indent=2, width=256) }} +{% endif %} +{% for key in kube_scheduler_config_extra_opts %} +{{ key }}: {{ kube_scheduler_config_extra_opts[key] }} +{% endfor %} diff --git a/kubespray/roles/kubernetes/control-plane/templates/podsecurity.yaml.j2 b/kubespray/roles/kubernetes/control-plane/templates/podsecurity.yaml.j2 new file mode 100644 index 0000000..c973733 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/podsecurity.yaml.j2 @@ -0,0 +1,17 @@ +{% if kube_pod_security_use_default %} +apiVersion: pod-security.admission.config.k8s.io/v1 +kind: PodSecurityConfiguration +defaults: + enforce: "{{ kube_pod_security_default_enforce }}" + enforce-version: "{{ kube_pod_security_default_enforce_version }}" + audit: "{{ kube_pod_security_default_audit }}" + audit-version: "{{ kube_pod_security_default_audit_version }}" + warn: "{{ kube_pod_security_default_warn }}" + warn-version: "{{ kube_pod_security_default_warn_version }}" +exemptions: + usernames: {{ kube_pod_security_exemptions_usernames | to_json }} + runtimeClasses: {{ kube_pod_security_exemptions_runtime_class_names | to_json }} + namespaces: {{ kube_pod_security_exemptions_namespaces | to_json }} +{% else %} +# This file is intentinally left empty as kube_pod_security_use_default={{ kube_pod_security_use_default }} +{% endif %} diff --git a/kubespray/roles/kubernetes/control-plane/templates/psp-cr.yml.j2 b/kubespray/roles/kubernetes/control-plane/templates/psp-cr.yml.j2 new file mode 100644 index 0000000..d9f0e8d --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/psp-cr.yml.j2 @@ -0,0 +1,32 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: psp:privileged + labels: + addonmanager.kubernetes.io/mode: Reconcile +rules: +- apiGroups: + - policy + resourceNames: + - privileged + resources: + - podsecuritypolicies + verbs: + - use +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: psp:restricted + labels: + addonmanager.kubernetes.io/mode: Reconcile +rules: +- apiGroups: + - policy + resourceNames: + - restricted + resources: + - podsecuritypolicies + verbs: + - use diff --git a/kubespray/roles/kubernetes/control-plane/templates/psp-crb.yml.j2 b/kubespray/roles/kubernetes/control-plane/templates/psp-crb.yml.j2 new file mode 100644 index 0000000..7513c3c --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/psp-crb.yml.j2 @@ -0,0 +1,54 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: psp:any:restricted +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: psp:restricted +subjects: +- kind: Group + name: system:authenticated + apiGroup: rbac.authorization.k8s.io +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: psp:kube-system:privileged + namespace: kube-system +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: psp:privileged +subjects: +- kind: Group + name: system:masters + apiGroup: rbac.authorization.k8s.io +- kind: Group + name: system:serviceaccounts:kube-system + apiGroup: rbac.authorization.k8s.io +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: psp:nodes:privileged + namespace: kube-system + annotations: + kubernetes.io/description: 'Allow nodes to create privileged pods. Should + be used in combination with the NodeRestriction admission plugin to limit + nodes to mirror pods bound to themselves.' + labels: + addonmanager.kubernetes.io/mode: Reconcile +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: psp:privileged +subjects: + - kind: Group + apiGroup: rbac.authorization.k8s.io + name: system:nodes + - kind: User + apiGroup: rbac.authorization.k8s.io + # Legacy node ID + name: kubelet diff --git a/kubespray/roles/kubernetes/control-plane/templates/psp.yml.j2 b/kubespray/roles/kubernetes/control-plane/templates/psp.yml.j2 new file mode 100644 index 0000000..5da5400 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/psp.yml.j2 @@ -0,0 +1,27 @@ +--- +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: restricted + annotations: + seccomp.security.alpha.kubernetes.io/defaultProfileName: 'runtime/default' + seccomp.security.alpha.kubernetes.io/allowedProfileNames: 'docker/default,runtime/default' +{% if apparmor_enabled %} + apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + apparmor.security.beta.kubernetes.io/allowedProfileNames: 'runtime/default' +{% endif %} + labels: + addonmanager.kubernetes.io/mode: Reconcile +spec: + {{ podsecuritypolicy_restricted_spec | to_yaml(indent=2, width=1337) | indent(width=2) }} +--- +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: privileged + annotations: + seccomp.security.alpha.kubernetes.io/allowedProfileNames: '*' + labels: + addonmanager.kubernetes.io/mode: Reconcile +spec: + {{ podsecuritypolicy_privileged_spec | to_yaml(indent=2, width=1337) | indent(width=2) }} diff --git a/kubespray/roles/kubernetes/control-plane/templates/secrets_encryption.yaml.j2 b/kubespray/roles/kubernetes/control-plane/templates/secrets_encryption.yaml.j2 new file mode 100644 index 0000000..3c521ff --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/secrets_encryption.yaml.j2 @@ -0,0 +1,11 @@ +apiVersion: apiserver.config.k8s.io/v1 +kind: EncryptionConfiguration +resources: + - resources: +{{ kube_encryption_resources | to_nice_yaml | indent(4, True) }} + providers: + - {{ kube_encryption_algorithm }}: + keys: + - name: key + secret: {{ kube_encrypt_token | b64encode }} + - identity: {} diff --git a/kubespray/roles/kubernetes/control-plane/templates/webhook-authorization-config.yaml.j2 b/kubespray/roles/kubernetes/control-plane/templates/webhook-authorization-config.yaml.j2 new file mode 100644 index 0000000..b5b5530 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/webhook-authorization-config.yaml.j2 @@ -0,0 +1,18 @@ +# clusters refers to the remote service. +clusters: +- name: webhook-token-authz-cluster + cluster: + server: {{ kube_webhook_authorization_url }} + insecure-skip-tls-verify: {{ kube_webhook_authorization_url_skip_tls_verify }} + +# users refers to the API server's webhook configuration. +users: +- name: webhook-token-authz-user + +# kubeconfig files require a context. Provide one for the API server. +current-context: webhook-token-authz +contexts: +- context: + cluster: webhook-token-authz-cluster + user: webhook-token-authz-user + name: webhook-token-authz diff --git a/kubespray/roles/kubernetes/control-plane/templates/webhook-token-auth-config.yaml.j2 b/kubespray/roles/kubernetes/control-plane/templates/webhook-token-auth-config.yaml.j2 new file mode 100644 index 0000000..f152d11 --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/templates/webhook-token-auth-config.yaml.j2 @@ -0,0 +1,21 @@ +# clusters refers to the remote service. +clusters: +- name: webhook-token-auth-cluster + cluster: + server: {{ kube_webhook_token_auth_url }} + insecure-skip-tls-verify: {{ kube_webhook_token_auth_url_skip_tls_verify }} +{% if kube_webhook_token_auth_ca_data is defined %} + certificate-authority-data: {{ kube_webhook_token_auth_ca_data }} +{% endif %} + +# users refers to the API server's webhook configuration. +users: +- name: webhook-token-auth-user + +# kubeconfig files require a context. Provide one for the API server. +current-context: webhook-token-auth +contexts: +- context: + cluster: webhook-token-auth-cluster + user: webhook-token-auth-user + name: webhook-token-auth diff --git a/kubespray/roles/kubernetes/control-plane/vars/main.yaml b/kubespray/roles/kubernetes/control-plane/vars/main.yaml new file mode 100644 index 0000000..f888d6b --- /dev/null +++ b/kubespray/roles/kubernetes/control-plane/vars/main.yaml @@ -0,0 +1,3 @@ +--- +# list of admission plugins that needs to be configured +kube_apiserver_admission_plugins_needs_configuration: [EventRateLimit, PodSecurity] diff --git a/kubespray/roles/kubernetes/kubeadm/defaults/main.yml b/kubespray/roles/kubernetes/kubeadm/defaults/main.yml new file mode 100644 index 0000000..61b132e --- /dev/null +++ b/kubespray/roles/kubernetes/kubeadm/defaults/main.yml @@ -0,0 +1,12 @@ +--- +# discovery_timeout modifies the discovery timeout +# This value must be smaller than kubeadm_join_timeout +discovery_timeout: 60s +kubeadm_join_timeout: 120s + +# If non-empty, will use this string as identification instead of the actual hostname +kube_override_hostname: >- + {%- if cloud_provider is defined and cloud_provider in ['aws'] -%} + {%- else -%} + {{ inventory_hostname }} + {%- endif -%} diff --git a/kubespray/roles/kubernetes/kubeadm/handlers/main.yml b/kubespray/roles/kubernetes/kubeadm/handlers/main.yml new file mode 100644 index 0000000..4c2b125 --- /dev/null +++ b/kubespray/roles/kubernetes/kubeadm/handlers/main.yml @@ -0,0 +1,15 @@ +--- +- name: Kubeadm | restart kubelet + command: /bin/true + notify: + - Kubeadm | reload systemd + - Kubeadm | reload kubelet + +- name: Kubeadm | reload systemd + systemd: + daemon_reload: true + +- name: Kubeadm | reload kubelet + service: + name: kubelet + state: restarted diff --git a/kubespray/roles/kubernetes/kubeadm/tasks/kubeadm_etcd_node.yml b/kubespray/roles/kubernetes/kubeadm/tasks/kubeadm_etcd_node.yml new file mode 100644 index 0000000..d39ea2b --- /dev/null +++ b/kubespray/roles/kubernetes/kubeadm/tasks/kubeadm_etcd_node.yml @@ -0,0 +1,62 @@ +--- +- name: Parse certificate key if not set + set_fact: + kubeadm_certificate_key: "{{ hostvars[groups['kube_control_plane'][0]]['kubeadm_certificate_key'] }}" + when: kubeadm_certificate_key is undefined + +- name: Create kubeadm cert controlplane config + template: + src: "kubeadm-client.conf.{{ kubeadmConfig_api_version }}.j2" + dest: "{{ kube_config_dir }}/kubeadm-cert-controlplane.conf" + mode: 0640 + vars: + kubeadm_cert_controlplane: true + +- name: Pull control plane certs down + shell: >- + {{ bin_dir }}/kubeadm join phase + control-plane-prepare download-certs + --config {{ kube_config_dir }}/kubeadm-cert-controlplane.conf + && + {{ bin_dir }}/kubeadm join phase + control-plane-prepare certs + --config {{ kube_config_dir }}/kubeadm-cert-controlplane.conf + args: + creates: "{{ kube_cert_dir }}/apiserver-etcd-client.key" + +- name: Delete unneeded certificates + file: + path: "{{ item }}" + state: absent + with_items: + - "{{ kube_cert_dir }}/apiserver.crt" + - "{{ kube_cert_dir }}/apiserver.key" + - "{{ kube_cert_dir }}/ca.key" + - "{{ kube_cert_dir }}/etcd/ca.key" + - "{{ kube_cert_dir }}/etcd/healthcheck-client.crt" + - "{{ kube_cert_dir }}/etcd/healthcheck-client.key" + - "{{ kube_cert_dir }}/etcd/peer.crt" + - "{{ kube_cert_dir }}/etcd/peer.key" + - "{{ kube_cert_dir }}/etcd/server.crt" + - "{{ kube_cert_dir }}/etcd/server.key" + - "{{ kube_cert_dir }}/front-proxy-ca.crt" + - "{{ kube_cert_dir }}/front-proxy-ca.key" + - "{{ kube_cert_dir }}/front-proxy-client.crt" + - "{{ kube_cert_dir }}/front-proxy-client.key" + - "{{ kube_cert_dir }}/sa.key" + - "{{ kube_cert_dir }}/sa.pub" + +- name: Calculate etcd cert serial + command: "openssl x509 -in {{ kube_cert_dir }}/apiserver-etcd-client.crt -noout -serial" + register: "etcd_client_cert_serial_result" + changed_when: false + when: + - inventory_hostname in groups['k8s_cluster'] | union(groups['calico_rr'] | default([])) | unique | sort + tags: + - network + +- name: Set etcd_client_cert_serial + set_fact: + etcd_client_cert_serial: "{{ etcd_client_cert_serial_result.stdout.split('=')[1] }}" + tags: + - network diff --git a/kubespray/roles/kubernetes/kubeadm/tasks/main.yml b/kubespray/roles/kubernetes/kubeadm/tasks/main.yml new file mode 100644 index 0000000..290eca3 --- /dev/null +++ b/kubespray/roles/kubernetes/kubeadm/tasks/main.yml @@ -0,0 +1,177 @@ +--- +- name: Set kubeadm_discovery_address + set_fact: + # noqa: jinja[spacing] + kubeadm_discovery_address: >- + {%- if "127.0.0.1" in kube_apiserver_endpoint or "localhost" in kube_apiserver_endpoint -%} + {{ first_kube_control_plane_address }}:{{ kube_apiserver_port }} + {%- else -%} + {{ kube_apiserver_endpoint | replace("https://", "") }} + {%- endif %} + tags: + - facts + +- name: Check if kubelet.conf exists + stat: + path: "{{ kube_config_dir }}/kubelet.conf" + get_attributes: no + get_checksum: no + get_mime: no + register: kubelet_conf + +- name: Check if kubeadm CA cert is accessible + stat: + path: "{{ kube_cert_dir }}/ca.crt" + get_attributes: no + get_checksum: no + get_mime: no + register: kubeadm_ca_stat + delegate_to: "{{ groups['kube_control_plane'][0] }}" + run_once: true + +- name: Calculate kubeadm CA cert hash + shell: set -o pipefail && openssl x509 -pubkey -in {{ kube_cert_dir }}/ca.crt | openssl rsa -pubin -outform der 2>/dev/null | openssl dgst -sha256 -hex | sed 's/^.* //' + args: + executable: /bin/bash + register: kubeadm_ca_hash + when: + - kubeadm_ca_stat.stat is defined + - kubeadm_ca_stat.stat.exists + delegate_to: "{{ groups['kube_control_plane'][0] }}" + run_once: true + changed_when: false + +- name: Create kubeadm token for joining nodes with 24h expiration (default) + command: "{{ bin_dir }}/kubeadm token create" + register: temp_token + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: kubeadm_token is not defined + changed_when: false + +- name: Set kubeadm_token to generated token + set_fact: + kubeadm_token: "{{ temp_token.stdout }}" + when: kubeadm_token is not defined + +- name: Set kubeadm api version to v1beta3 + set_fact: + kubeadmConfig_api_version: v1beta3 + +- name: Create kubeadm client config + template: + src: "kubeadm-client.conf.{{ kubeadmConfig_api_version }}.j2" + dest: "{{ kube_config_dir }}/kubeadm-client.conf" + backup: yes + mode: 0640 + when: not is_kube_master + +- name: Kubeadm | Create directory to store kubeadm patches + file: + path: "{{ kubeadm_patches.dest_dir }}" + state: directory + mode: 0640 + when: kubeadm_patches is defined and kubeadm_patches.enabled + +- name: Kubeadm | Copy kubeadm patches from inventory files + copy: + src: "{{ kubeadm_patches.source_dir }}/" + dest: "{{ kubeadm_patches.dest_dir }}" + owner: "root" + mode: 0644 + when: kubeadm_patches is defined and kubeadm_patches.enabled + +- name: Join to cluster if needed + environment: + PATH: "{{ bin_dir }}:{{ ansible_env.PATH }}:/sbin" + when: not is_kube_master and (not kubelet_conf.stat.exists) + block: + + - name: Join to cluster + command: >- + timeout -k {{ kubeadm_join_timeout }} {{ kubeadm_join_timeout }} + {{ bin_dir }}/kubeadm join + --config {{ kube_config_dir }}/kubeadm-client.conf + --ignore-preflight-errors=DirAvailable--etc-kubernetes-manifests + --skip-phases={{ kubeadm_join_phases_skip | join(',') }} + register: kubeadm_join + changed_when: kubeadm_join is success + + rescue: + + - name: Join to cluster with ignores + command: >- + timeout -k {{ kubeadm_join_timeout }} {{ kubeadm_join_timeout }} + {{ bin_dir }}/kubeadm join + --config {{ kube_config_dir }}/kubeadm-client.conf + --ignore-preflight-errors=all + --skip-phases={{ kubeadm_join_phases_skip | join(',') }} + register: kubeadm_join + changed_when: kubeadm_join is success + + always: + + - name: Display kubeadm join stderr if any + when: kubeadm_join is failed + debug: + msg: | + Joined with warnings + {{ kubeadm_join.stderr_lines }} + +- name: Update server field in kubelet kubeconfig + lineinfile: + dest: "{{ kube_config_dir }}/kubelet.conf" + regexp: 'server:' + line: ' server: {{ kube_apiserver_endpoint }}' + backup: yes + when: + - kubeadm_config_api_fqdn is not defined + - not is_kube_master + - kubeadm_discovery_address != kube_apiserver_endpoint | replace("https://", "") + notify: Kubeadm | restart kubelet + +# FIXME(mattymo): Need to point to localhost, otherwise masters will all point +# incorrectly to first master, creating SPoF. +- name: Update server field in kube-proxy kubeconfig + shell: >- + set -o pipefail && {{ kubectl }} get configmap kube-proxy -n kube-system -o yaml + | sed 's#server:.*#server: https://127.0.0.1:{{ kube_apiserver_port }}#g' + | {{ kubectl }} replace -f - + args: + executable: /bin/bash + run_once: true + delegate_to: "{{ groups['kube_control_plane'] | first }}" + delegate_facts: false + when: + - kubeadm_config_api_fqdn is not defined + - kubeadm_discovery_address != kube_apiserver_endpoint | replace("https://", "") + - kube_proxy_deployed + - loadbalancer_apiserver_localhost + tags: + - kube-proxy + +- name: Set ca.crt file permission + file: + path: "{{ kube_cert_dir }}/ca.crt" + owner: root + group: root + mode: "0644" + +- name: Restart all kube-proxy pods to ensure that they load the new configmap + command: "{{ kubectl }} delete pod -n kube-system -l k8s-app=kube-proxy --force --grace-period=0" + run_once: true + delegate_to: "{{ groups['kube_control_plane'] | first }}" + delegate_facts: false + when: + - kubeadm_config_api_fqdn is not defined + - kubeadm_discovery_address != kube_apiserver_endpoint | replace("https://", "") + - kube_proxy_deployed + tags: + - kube-proxy + +- name: Extract etcd certs from control plane if using etcd kubeadm mode + include_tasks: kubeadm_etcd_node.yml + when: + - etcd_deployment_type == "kubeadm" + - inventory_hostname not in groups['kube_control_plane'] + - kube_network_plugin in ["calico", "flannel", "cilium"] or cilium_deploy_additionally | default(false) | bool + - kube_network_plugin != "calico" or calico_datastore == "etcd" diff --git a/kubespray/roles/kubernetes/kubeadm/templates/kubeadm-client.conf.v1beta3.j2 b/kubespray/roles/kubernetes/kubeadm/templates/kubeadm-client.conf.v1beta3.j2 new file mode 100644 index 0000000..5104ecf --- /dev/null +++ b/kubespray/roles/kubernetes/kubeadm/templates/kubeadm-client.conf.v1beta3.j2 @@ -0,0 +1,39 @@ +--- +apiVersion: kubeadm.k8s.io/v1beta3 +kind: JoinConfiguration +discovery: + bootstrapToken: +{% if kubeadm_config_api_fqdn is defined %} + apiServerEndpoint: {{ kubeadm_config_api_fqdn }}:{{ loadbalancer_apiserver.port | default(kube_apiserver_port) }} +{% else %} + apiServerEndpoint: {{ kubeadm_discovery_address }} +{% endif %} + token: {{ kubeadm_token }} +{% if kubeadm_ca_hash.stdout is defined %} + caCertHashes: + - sha256:{{ kubeadm_ca_hash.stdout }} +{% else %} + unsafeSkipCAVerification: true +{% endif %} + timeout: {{ discovery_timeout }} + tlsBootstrapToken: {{ kubeadm_token }} +caCertPath: {{ kube_cert_dir }}/ca.crt +{% if kubeadm_cert_controlplane is defined and kubeadm_cert_controlplane %} +controlPlane: + localAPIEndpoint: + advertiseAddress: {{ kube_apiserver_address }} + bindPort: {{ kube_apiserver_port }} + certificateKey: {{ kubeadm_certificate_key }} +{% endif %} +nodeRegistration: + name: '{{ kube_override_hostname }}' + criSocket: {{ cri_socket }} +{% if 'calico_rr' in group_names and 'kube_node' not in group_names %} + taints: + - effect: NoSchedule + key: node-role.kubernetes.io/calico-rr +{% endif %} +{% if kubeadm_patches is defined and kubeadm_patches.enabled %} +patches: + directory: {{ kubeadm_patches.dest_dir }} +{% endif %} diff --git a/kubespray/roles/kubernetes/node-label/tasks/main.yml b/kubespray/roles/kubernetes/node-label/tasks/main.yml new file mode 100644 index 0000000..cda700c --- /dev/null +++ b/kubespray/roles/kubernetes/node-label/tasks/main.yml @@ -0,0 +1,49 @@ +--- +- name: Kubernetes Apps | Wait for kube-apiserver + uri: + url: "{{ kube_apiserver_endpoint }}/healthz" + validate_certs: no + client_cert: "{{ kube_apiserver_client_cert }}" + client_key: "{{ kube_apiserver_client_key }}" + register: result + until: result.status == 200 + retries: 10 + delay: 6 + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Set role node label to empty list + set_fact: + role_node_labels: [] + +- name: Node label for nvidia GPU nodes + set_fact: + role_node_labels: "{{ role_node_labels + ['nvidia.com/gpu=true'] }}" + when: + - nvidia_gpu_nodes is defined + - nvidia_accelerator_enabled | bool + - inventory_hostname in nvidia_gpu_nodes + +- name: Set inventory node label to empty list + set_fact: + inventory_node_labels: [] + +- name: Populate inventory node label + set_fact: + inventory_node_labels: "{{ inventory_node_labels + ['%s=%s' | format(item.key, item.value)] }}" + loop: "{{ node_labels | d({}) | dict2items }}" + when: + - node_labels is defined + - node_labels is mapping + +- debug: # noqa name[missing] + var: role_node_labels +- debug: # noqa name[missing] + var: inventory_node_labels + +- name: Set label to node + command: >- + {{ kubectl }} label node {{ kube_override_hostname | default(inventory_hostname) }} {{ item }} --overwrite=true + loop: "{{ role_node_labels + inventory_node_labels }}" + delegate_to: "{{ groups['kube_control_plane'][0] }}" + changed_when: false +... diff --git a/kubespray/roles/kubernetes/node/defaults/main.yml b/kubespray/roles/kubernetes/node/defaults/main.yml new file mode 100644 index 0000000..f5dbf38 --- /dev/null +++ b/kubespray/roles/kubernetes/node/defaults/main.yml @@ -0,0 +1,242 @@ +--- +# advertised host IP for kubelet. This affects network plugin config. Take caution +kubelet_address: "{{ ip | default(fallback_ips[inventory_hostname]) }}{{ (',' + ip6) if enable_dual_stack_networks and ip6 is defined else '' }}" + +# bind address for kubelet. Set to 0.0.0.0 to listen on all interfaces +kubelet_bind_address: "{{ ip | default('0.0.0.0') }}" + +# resolv.conf to base dns config +kube_resolv_conf: "/etc/resolv.conf" + +# Set to empty to avoid cgroup creation +kubelet_enforce_node_allocatable: "\"\"" + +# Set runtime and kubelet cgroups when using systemd as cgroup driver (default) +kube_service_cgroups: "{% if kube_reserved %}{{ kube_reserved_cgroups_for_service_slice }}{% else %}system.slice{% endif %}" +kubelet_runtime_cgroups: "/{{ kube_service_cgroups }}/{{ container_manager }}.service" +kubelet_kubelet_cgroups: "/{{ kube_service_cgroups }}/kubelet.service" + +# Set runtime and kubelet cgroups when using cgroupfs as cgroup driver +kubelet_runtime_cgroups_cgroupfs: "/system.slice/{{ container_manager }}.service" +kubelet_kubelet_cgroups_cgroupfs: "/system.slice/kubelet.service" + +# Set systemd service hardening features +kubelet_systemd_hardening: false + +# List of secure IPs for kubelet +kubelet_secure_addresses: >- + {%- for host in groups['kube_control_plane'] -%} + {{ hostvars[host]['ip'] | default(fallback_ips[host]) }}{{ ' ' if not loop.last else '' }} + {%- endfor -%} + +# Reserve this space for kube resources +# Set to true to reserve resources for kube daemons +kube_reserved: false +kube_reserved_cgroups_for_service_slice: kube.slice +kube_reserved_cgroups: "/{{ kube_reserved_cgroups_for_service_slice }}" +kube_memory_reserved: 256Mi +kube_cpu_reserved: 100m +# kube_ephemeral_storage_reserved: 2Gi +# kube_pid_reserved: "1000" +# Reservation for master hosts +kube_master_memory_reserved: 512Mi +kube_master_cpu_reserved: 200m +# kube_master_ephemeral_storage_reserved: 2Gi +# kube_master_pid_reserved: "1000" + +# Set to true to reserve resources for system daemons +system_reserved: false +system_reserved_cgroups_for_service_slice: system.slice +system_reserved_cgroups: "/{{ system_reserved_cgroups_for_service_slice }}" +system_memory_reserved: 512Mi +system_cpu_reserved: 500m +# system_ephemeral_storage_reserved: 2Gi +# system_pid_reserved: "1000" +# Reservation for master hosts +system_master_memory_reserved: 256Mi +system_master_cpu_reserved: 250m +# system_master_ephemeral_storage_reserved: 2Gi +# system_master_pid_reserved: "1000" + +## Eviction Thresholds to avoid system OOMs +# https://kubernetes.io/docs/tasks/administer-cluster/reserve-compute-resources/#eviction-thresholds +eviction_hard: {} +eviction_hard_control_plane: {} + +kubelet_status_update_frequency: 10s + +# kube-vip +kube_vip_version: v0.5.12 + +kube_vip_arp_enabled: false +kube_vip_interface: +kube_vip_services_interface: +kube_vip_cidr: 32 +kube_vip_controlplane_enabled: false +kube_vip_ddns_enabled: false +kube_vip_services_enabled: false +kube_vip_leader_election_enabled: "{{ kube_vip_arp_enabled }}" +kube_vip_bgp_enabled: false +kube_vip_bgp_routerid: +kube_vip_local_as: 65000 +kube_vip_bgp_peeraddress: +kube_vip_bgp_peerpass: +kube_vip_bgp_peeras: 65000 +kube_vip_bgppeers: +kube_vip_address: +kube_vip_enableServicesElection: false +kube_vip_lb_enable: false + +# Requests for load balancer app +loadbalancer_apiserver_memory_requests: 32M +loadbalancer_apiserver_cpu_requests: 25m + +loadbalancer_apiserver_keepalive_timeout: 5m +loadbalancer_apiserver_pod_name: "{% if loadbalancer_apiserver_type == 'nginx' %}nginx-proxy{% else %}haproxy{% endif %}" + +# Uncomment if you need to enable deprecated runtimes +# kube_api_runtime_config: +# - apps/v1beta1=true +# - apps/v1beta2=true +# - extensions/v1beta1/daemonsets=true +# - extensions/v1beta1/deployments=true +# - extensions/v1beta1/replicasets=true +# - extensions/v1beta1/networkpolicies=true +# - extensions/v1beta1/podsecuritypolicies=true + +# A port range to reserve for services with NodePort visibility. +# Inclusive at both ends of the range. +kube_apiserver_node_port_range: "30000-32767" + +# Configure the amount of pods able to run on single node +# default is equal to application default +kubelet_max_pods: 110 + +# Sets the maximum number of processes running per Pod +# Default value -1 = unlimited +kubelet_pod_pids_limit: -1 + +## Support parameters to be passed to kubelet via kubelet-config.yaml +kubelet_config_extra_args: {} + +## Parameters to be passed to kubelet via kubelet-config.yaml when cgroupfs is used as cgroup driver +kubelet_config_extra_args_cgroupfs: + systemCgroups: /system.slice + cgroupRoot: / + +## Support parameters to be passed to kubelet via kubelet-config.yaml only on nodes, not masters +kubelet_node_config_extra_args: {} + +# Maximum number of container log files that can be present for a container. +kubelet_logfiles_max_nr: 5 + +# Maximum size of the container log file before it is rotated +kubelet_logfiles_max_size: 10Mi + +## Support custom flags to be passed to kubelet +kubelet_custom_flags: [] + +## Support custom flags to be passed to kubelet only on nodes, not masters +kubelet_node_custom_flags: [] + +# If non-empty, will use this string as identification instead of the actual hostname +kube_override_hostname: >- + {%- if cloud_provider is defined and cloud_provider in ['aws'] -%} + {%- else -%} + {{ inventory_hostname }} + {%- endif -%} + +# The read-only port for the Kubelet to serve on with no authentication/authorization. +kube_read_only_port: 0 + +# Port for healthz for Kubelet +kubelet_healthz_port: 10248 + +# Bind address for healthz for Kubelet +kubelet_healthz_bind_address: 127.0.0.1 + +# sysctl_file_path to add sysctl conf to +sysctl_file_path: "/etc/sysctl.d/99-sysctl.conf" + +# For the openstack integration kubelet will need credentials to access +# openstack apis like nova and cinder. Per default this values will be +# read from the environment. +openstack_auth_url: "{{ lookup('env', 'OS_AUTH_URL') }}" +openstack_username: "{{ lookup('env', 'OS_USERNAME') }}" +openstack_password: "{{ lookup('env', 'OS_PASSWORD') }}" +openstack_region: "{{ lookup('env', 'OS_REGION_NAME') }}" +openstack_tenant_id: "{{ lookup('env', 'OS_TENANT_ID') | default(lookup('env', 'OS_PROJECT_ID') | default(lookup('env', 'OS_PROJECT_NAME'), true), true) }}" +openstack_tenant_name: "{{ lookup('env', 'OS_TENANT_NAME') }}" +openstack_domain_name: "{{ lookup('env', 'OS_USER_DOMAIN_NAME') }}" +openstack_domain_id: "{{ lookup('env', 'OS_USER_DOMAIN_ID') }}" + +# For the vsphere integration, kubelet will need credentials to access +# vsphere apis +# Documentation regarding these values can be found +# https://github.com/kubernetes/kubernetes/blob/master/pkg/cloudprovider/providers/vsphere/vsphere.go#L105 +vsphere_vcenter_ip: "{{ lookup('env', 'VSPHERE_VCENTER') }}" +vsphere_vcenter_port: "{{ lookup('env', 'VSPHERE_VCENTER_PORT') }}" +vsphere_user: "{{ lookup('env', 'VSPHERE_USER') }}" +vsphere_password: "{{ lookup('env', 'VSPHERE_PASSWORD') }}" +vsphere_datacenter: "{{ lookup('env', 'VSPHERE_DATACENTER') }}" +vsphere_datastore: "{{ lookup('env', 'VSPHERE_DATASTORE') }}" +vsphere_working_dir: "{{ lookup('env', 'VSPHERE_WORKING_DIR') }}" +vsphere_insecure: "{{ lookup('env', 'VSPHERE_INSECURE') }}" +vsphere_resource_pool: "{{ lookup('env', 'VSPHERE_RESOURCE_POOL') }}" + +vsphere_scsi_controller_type: pvscsi +# vsphere_public_network is name of the network the VMs are joined to +vsphere_public_network: "{{ lookup('env', 'VSPHERE_PUBLIC_NETWORK') | default('') }}" + +## When azure is used, you need to also set the following variables. +## see docs/azure.md for details on how to get these values +# azure_tenant_id: +# azure_subscription_id: +# azure_aad_client_id: +# azure_aad_client_secret: +# azure_resource_group: +# azure_location: +# azure_subnet_name: +# azure_security_group_name: +# azure_vnet_name: +# azure_route_table_name: +# supported values are 'standard' or 'vmss' +# azure_vmtype: standard +# Sku of Load Balancer and Public IP. Candidate values are: basic and standard. +azure_loadbalancer_sku: basic +# excludes master nodes from standard load balancer. +azure_exclude_master_from_standard_lb: true +# disables the outbound SNAT for public load balancer rules +azure_disable_outbound_snat: false +# use instance metadata service where possible +azure_use_instance_metadata: true +# use specific Azure API endpoints +azure_cloud: AzurePublicCloud + +## Support tls min version, Possible values: VersionTLS10, VersionTLS11, VersionTLS12, VersionTLS13. +# tls_min_version: "" + +## Support tls cipher suites. +# tls_cipher_suites: +# - TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA +# - TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256 +# - TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 +# - TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA +# - TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 +# - TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305 +# - TLS_ECDHE_ECDSA_WITH_RC4_128_SHA +# - TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256 +# - TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 +# - TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA +# - TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 +# - TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305 +# - TLS_ECDHE_RSA_WITH_RC4_128_SHA +# - TLS_RSA_WITH_3DES_EDE_CBC_SHA +# - TLS_RSA_WITH_AES_128_CBC_SHA +# - TLS_RSA_WITH_AES_128_CBC_SHA256 +# - TLS_RSA_WITH_AES_128_GCM_SHA256 +# - TLS_RSA_WITH_AES_256_CBC_SHA +# - TLS_RSA_WITH_AES_256_GCM_SHA384 +# - TLS_RSA_WITH_RC4_128_SHA diff --git a/kubespray/roles/kubernetes/node/handlers/main.yml b/kubespray/roles/kubernetes/node/handlers/main.yml new file mode 100644 index 0000000..512b4e8 --- /dev/null +++ b/kubespray/roles/kubernetes/node/handlers/main.yml @@ -0,0 +1,15 @@ +--- +- name: Node | restart kubelet + command: /bin/true + notify: + - Kubelet | reload systemd + - Kubelet | restart kubelet + +- name: Kubelet | reload systemd + systemd: + daemon_reload: true + +- name: Kubelet | restart kubelet + service: + name: kubelet + state: restarted diff --git a/kubespray/roles/kubernetes/node/tasks/cloud-credentials/azure-credential-check.yml b/kubespray/roles/kubernetes/node/tasks/cloud-credentials/azure-credential-check.yml new file mode 100644 index 0000000..c5d6030 --- /dev/null +++ b/kubespray/roles/kubernetes/node/tasks/cloud-credentials/azure-credential-check.yml @@ -0,0 +1,82 @@ +--- +- name: Check azure_tenant_id value + fail: + msg: "azure_tenant_id is missing" + when: azure_tenant_id is not defined or not azure_tenant_id + +- name: Check azure_subscription_id value + fail: + msg: "azure_subscription_id is missing" + when: azure_subscription_id is not defined or not azure_subscription_id + +- name: Check azure_aad_client_id value + fail: + msg: "azure_aad_client_id is missing" + when: azure_aad_client_id is not defined or not azure_aad_client_id + +- name: Check azure_aad_client_secret value + fail: + msg: "azure_aad_client_secret is missing" + when: azure_aad_client_secret is not defined or not azure_aad_client_secret + +- name: Check azure_resource_group value + fail: + msg: "azure_resource_group is missing" + when: azure_resource_group is not defined or not azure_resource_group + +- name: Check azure_location value + fail: + msg: "azure_location is missing" + when: azure_location is not defined or not azure_location + +- name: Check azure_subnet_name value + fail: + msg: "azure_subnet_name is missing" + when: azure_subnet_name is not defined or not azure_subnet_name + +- name: Check azure_security_group_name value + fail: + msg: "azure_security_group_name is missing" + when: azure_security_group_name is not defined or not azure_security_group_name + +- name: Check azure_vnet_name value + fail: + msg: "azure_vnet_name is missing" + when: azure_vnet_name is not defined or not azure_vnet_name + +- name: Check azure_vnet_resource_group value + fail: + msg: "azure_vnet_resource_group is missing" + when: azure_vnet_resource_group is not defined or not azure_vnet_resource_group + +- name: Check azure_route_table_name value + fail: + msg: "azure_route_table_name is missing" + when: azure_route_table_name is not defined or not azure_route_table_name + +- name: Check azure_loadbalancer_sku value + fail: + msg: "azure_loadbalancer_sku has an invalid value '{{ azure_loadbalancer_sku }}'. Supported values are 'basic', 'standard'" + when: azure_loadbalancer_sku not in ["basic", "standard"] + +- name: "Check azure_exclude_master_from_standard_lb is a bool" + assert: + that: azure_exclude_master_from_standard_lb | type_debug == 'bool' + +- name: "Check azure_disable_outbound_snat is a bool" + assert: + that: azure_disable_outbound_snat | type_debug == 'bool' + +- name: "Check azure_use_instance_metadata is a bool" + assert: + that: azure_use_instance_metadata | type_debug == 'bool' + +- name: Check azure_vmtype value + fail: + msg: "azure_vmtype is missing. Supported values are 'standard' or 'vmss'" + when: azure_vmtype is not defined or not azure_vmtype + +- name: Check azure_cloud value + fail: + msg: "azure_cloud has an invalid value '{{ azure_cloud }}'. Supported values are 'AzureChinaCloud', 'AzureGermanCloud', 'AzurePublicCloud', 'AzureUSGovernmentCloud'." + when: azure_cloud not in ["AzureChinaCloud", "AzureGermanCloud", "AzurePublicCloud", "AzureUSGovernmentCloud"] diff --git a/kubespray/roles/kubernetes/node/tasks/cloud-credentials/openstack-credential-check.yml b/kubespray/roles/kubernetes/node/tasks/cloud-credentials/openstack-credential-check.yml new file mode 100644 index 0000000..7354d43 --- /dev/null +++ b/kubespray/roles/kubernetes/node/tasks/cloud-credentials/openstack-credential-check.yml @@ -0,0 +1,34 @@ +--- +- name: Check openstack_auth_url value + fail: + msg: "openstack_auth_url is missing" + when: openstack_auth_url is not defined or not openstack_auth_url + +- name: Check openstack_username value + fail: + msg: "openstack_username is missing" + when: openstack_username is not defined or not openstack_username + +- name: Check openstack_password value + fail: + msg: "openstack_password is missing" + when: openstack_password is not defined or not openstack_password + +- name: Check openstack_region value + fail: + msg: "openstack_region is missing" + when: openstack_region is not defined or not openstack_region + +- name: Check openstack_tenant_id value + fail: + msg: "one of openstack_tenant_id or openstack_trust_id must be specified" + when: + - openstack_tenant_id is not defined or not openstack_tenant_id + - openstack_trust_id is not defined + +- name: Check openstack_trust_id value + fail: + msg: "one of openstack_tenant_id or openstack_trust_id must be specified" + when: + - openstack_trust_id is not defined or not openstack_trust_id + - openstack_tenant_id is not defined diff --git a/kubespray/roles/kubernetes/node/tasks/cloud-credentials/vsphere-credential-check.yml b/kubespray/roles/kubernetes/node/tasks/cloud-credentials/vsphere-credential-check.yml new file mode 100644 index 0000000..b18583a --- /dev/null +++ b/kubespray/roles/kubernetes/node/tasks/cloud-credentials/vsphere-credential-check.yml @@ -0,0 +1,22 @@ +--- +- name: Check vsphere environment variables + fail: + msg: "{{ item.name }} is missing" + when: item.value is not defined or not item.value + with_items: + - name: vsphere_vcenter_ip + value: "{{ vsphere_vcenter_ip }}" + - name: vsphere_vcenter_port + value: "{{ vsphere_vcenter_port }}" + - name: vsphere_user + value: "{{ vsphere_user }}" + - name: vsphere_password + value: "{{ vsphere_password }}" + - name: vsphere_datacenter + value: "{{ vsphere_datacenter }}" + - name: vsphere_datastore + value: "{{ vsphere_datastore }}" + - name: vsphere_working_dir + value: "{{ vsphere_working_dir }}" + - name: vsphere_insecure + value: "{{ vsphere_insecure }}" diff --git a/kubespray/roles/kubernetes/node/tasks/facts.yml b/kubespray/roles/kubernetes/node/tasks/facts.yml new file mode 100644 index 0000000..156d748 --- /dev/null +++ b/kubespray/roles/kubernetes/node/tasks/facts.yml @@ -0,0 +1,62 @@ +--- +- name: Gather cgroups facts for docker + when: container_manager == 'docker' + block: + - name: Look up docker cgroup driver + shell: "set -o pipefail && docker info | grep 'Cgroup Driver' | awk -F': ' '{ print $2; }'" + args: + executable: /bin/bash + register: docker_cgroup_driver_result + changed_when: false + check_mode: no + + - name: Set kubelet_cgroup_driver_detected fact for docker + set_fact: + kubelet_cgroup_driver_detected: "{{ docker_cgroup_driver_result.stdout }}" + +- name: Gather cgroups facts for crio + when: container_manager == 'crio' + block: + - name: Look up crio cgroup driver + shell: "set -o pipefail && {{ bin_dir }}/crio-status info | grep 'cgroup driver' | awk -F': ' '{ print $2; }'" + args: + executable: /bin/bash + register: crio_cgroup_driver_result + changed_when: false + + - name: Set kubelet_cgroup_driver_detected fact for crio + set_fact: + kubelet_cgroup_driver_detected: "{{ crio_cgroup_driver_result.stdout }}" + +- name: Set kubelet_cgroup_driver_detected fact for containerd + when: container_manager == 'containerd' + set_fact: + kubelet_cgroup_driver_detected: >- + {%- if containerd_use_systemd_cgroup -%}systemd{%- else -%}cgroupfs{%- endif -%} + +- name: Set kubelet_cgroup_driver + set_fact: + kubelet_cgroup_driver: "{{ kubelet_cgroup_driver_detected }}" + when: kubelet_cgroup_driver is undefined + +- name: Set kubelet_cgroups options when cgroupfs is used + set_fact: + kubelet_runtime_cgroups: "{{ kubelet_runtime_cgroups_cgroupfs }}" + kubelet_kubelet_cgroups: "{{ kubelet_kubelet_cgroups_cgroupfs }}" + when: kubelet_cgroup_driver == 'cgroupfs' + +- name: Set kubelet_config_extra_args options when cgroupfs is used + set_fact: + kubelet_config_extra_args: "{{ kubelet_config_extra_args | combine(kubelet_config_extra_args_cgroupfs) }}" + when: kubelet_cgroup_driver == 'cgroupfs' + +- name: Os specific vars + include_vars: "{{ item }}" + with_first_found: + - files: + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_release }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_major_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}.yml" + - "{{ ansible_os_family | lower }}.yml" + skip: true diff --git a/kubespray/roles/kubernetes/node/tasks/install.yml b/kubespray/roles/kubernetes/node/tasks/install.yml new file mode 100644 index 0000000..fb1e8ad --- /dev/null +++ b/kubespray/roles/kubernetes/node/tasks/install.yml @@ -0,0 +1,22 @@ +--- +- name: Install | Copy kubeadm binary from download dir + copy: + src: "{{ downloads.kubeadm.dest }}" + dest: "{{ bin_dir }}/kubeadm" + mode: 0755 + remote_src: true + tags: + - kubeadm + when: + - not inventory_hostname in groups['kube_control_plane'] + +- name: Install | Copy kubelet binary from download dir + copy: + src: "{{ downloads.kubelet.dest }}" + dest: "{{ bin_dir }}/kubelet" + mode: 0755 + remote_src: true + tags: + - kubelet + - upgrade + notify: Node | restart kubelet diff --git a/kubespray/roles/kubernetes/node/tasks/kubelet.yml b/kubespray/roles/kubernetes/node/tasks/kubelet.yml new file mode 100644 index 0000000..ee01d06 --- /dev/null +++ b/kubespray/roles/kubernetes/node/tasks/kubelet.yml @@ -0,0 +1,52 @@ +--- +- name: Set kubelet api version to v1beta1 + set_fact: + kubeletConfig_api_version: v1beta1 + tags: + - kubelet + - kubeadm + +- name: Write kubelet environment config file (kubeadm) + template: + src: "kubelet.env.{{ kubeletConfig_api_version }}.j2" + dest: "{{ kube_config_dir }}/kubelet.env" + setype: "{{ (preinstall_selinux_state != 'disabled') | ternary('etc_t', omit) }}" + backup: yes + mode: 0600 + notify: Node | restart kubelet + tags: + - kubelet + - kubeadm + +- name: Write kubelet config file + template: + src: "kubelet-config.{{ kubeletConfig_api_version }}.yaml.j2" + dest: "{{ kube_config_dir }}/kubelet-config.yaml" + mode: 0600 + notify: Kubelet | restart kubelet + tags: + - kubelet + - kubeadm + +- name: Write kubelet systemd init file + template: + src: "kubelet.service.j2" + dest: "/etc/systemd/system/kubelet.service" + backup: "yes" + mode: 0600 + notify: Node | restart kubelet + tags: + - kubelet + - kubeadm + +- name: Flush_handlers and reload-systemd + meta: flush_handlers + +- name: Enable kubelet + service: + name: kubelet + enabled: yes + state: started + tags: + - kubelet + notify: Kubelet | restart kubelet diff --git a/kubespray/roles/kubernetes/node/tasks/loadbalancer/haproxy.yml b/kubespray/roles/kubernetes/node/tasks/loadbalancer/haproxy.yml new file mode 100644 index 0000000..7e5cfce --- /dev/null +++ b/kubespray/roles/kubernetes/node/tasks/loadbalancer/haproxy.yml @@ -0,0 +1,34 @@ +--- +- name: Haproxy | Cleanup potentially deployed nginx-proxy + file: + path: "{{ kube_manifest_dir }}/nginx-proxy.yml" + state: absent + +- name: Haproxy | Make haproxy directory + file: + path: "{{ haproxy_config_dir }}" + state: directory + mode: 0755 + owner: root + +- name: Haproxy | Write haproxy configuration + template: + src: "loadbalancer/haproxy.cfg.j2" + dest: "{{ haproxy_config_dir }}/haproxy.cfg" + owner: root + mode: 0755 + backup: yes + +- name: Haproxy | Get checksum from config + stat: + path: "{{ haproxy_config_dir }}/haproxy.cfg" + get_attributes: no + get_checksum: yes + get_mime: no + register: haproxy_stat + +- name: Haproxy | Write static pod + template: + src: manifests/haproxy.manifest.j2 + dest: "{{ kube_manifest_dir }}/haproxy.yml" + mode: 0640 diff --git a/kubespray/roles/kubernetes/node/tasks/loadbalancer/kube-vip.yml b/kubespray/roles/kubernetes/node/tasks/loadbalancer/kube-vip.yml new file mode 100644 index 0000000..f7b04a6 --- /dev/null +++ b/kubespray/roles/kubernetes/node/tasks/loadbalancer/kube-vip.yml @@ -0,0 +1,13 @@ +--- +- name: Kube-vip | Check cluster settings for kube-vip + fail: + msg: "kube-vip require kube_proxy_strict_arp = true, see https://github.com/kube-vip/kube-vip/blob/main/docs/kubernetes/arp/index.md" + when: + - kube_proxy_mode == 'ipvs' and not kube_proxy_strict_arp + - kube_vip_arp_enabled + +- name: Kube-vip | Write static pod + template: + src: manifests/kube-vip.manifest.j2 + dest: "{{ kube_manifest_dir }}/kube-vip.yml" + mode: 0640 diff --git a/kubespray/roles/kubernetes/node/tasks/loadbalancer/nginx-proxy.yml b/kubespray/roles/kubernetes/node/tasks/loadbalancer/nginx-proxy.yml new file mode 100644 index 0000000..5b82ff6 --- /dev/null +++ b/kubespray/roles/kubernetes/node/tasks/loadbalancer/nginx-proxy.yml @@ -0,0 +1,34 @@ +--- +- name: Haproxy | Cleanup potentially deployed haproxy + file: + path: "{{ kube_manifest_dir }}/haproxy.yml" + state: absent + +- name: Nginx-proxy | Make nginx directory + file: + path: "{{ nginx_config_dir }}" + state: directory + mode: 0700 + owner: root + +- name: Nginx-proxy | Write nginx-proxy configuration + template: + src: "loadbalancer/nginx.conf.j2" + dest: "{{ nginx_config_dir }}/nginx.conf" + owner: root + mode: 0755 + backup: yes + +- name: Nginx-proxy | Get checksum from config + stat: + path: "{{ nginx_config_dir }}/nginx.conf" + get_attributes: no + get_checksum: yes + get_mime: no + register: nginx_stat + +- name: Nginx-proxy | Write static pod + template: + src: manifests/nginx-proxy.manifest.j2 + dest: "{{ kube_manifest_dir }}/nginx-proxy.yml" + mode: 0640 diff --git a/kubespray/roles/kubernetes/node/tasks/main.yml b/kubespray/roles/kubernetes/node/tasks/main.yml new file mode 100644 index 0000000..f89e03e --- /dev/null +++ b/kubespray/roles/kubernetes/node/tasks/main.yml @@ -0,0 +1,202 @@ +--- +- name: Fetch facts + import_tasks: facts.yml + tags: + - facts + - kubelet + +- name: Pre-upgrade kubelet + import_tasks: pre_upgrade.yml + tags: + - kubelet + +- name: Ensure /var/lib/cni exists + file: + path: /var/lib/cni + state: directory + mode: 0755 + +- name: Install kubelet binary + import_tasks: install.yml + tags: + - kubelet + +- name: Install kube-vip + import_tasks: loadbalancer/kube-vip.yml + when: + - is_kube_master + - kube_vip_enabled + tags: + - kube-vip + +- name: Install nginx-proxy + import_tasks: loadbalancer/nginx-proxy.yml + when: + - not is_kube_master or kube_apiserver_bind_address != '0.0.0.0' + - loadbalancer_apiserver_localhost + - loadbalancer_apiserver_type == 'nginx' + tags: + - nginx + +- name: Install haproxy + import_tasks: loadbalancer/haproxy.yml + when: + - not is_kube_master or kube_apiserver_bind_address != '0.0.0.0' + - loadbalancer_apiserver_localhost + - loadbalancer_apiserver_type == 'haproxy' + tags: + - haproxy + +- name: Ensure nodePort range is reserved + ansible.posix.sysctl: + name: net.ipv4.ip_local_reserved_ports + value: "{{ kube_apiserver_node_port_range }}" + sysctl_set: yes + sysctl_file: "{{ sysctl_file_path }}" + state: present + reload: yes + when: kube_apiserver_node_port_range is defined + tags: + - kube-proxy + +- name: Verify if br_netfilter module exists + command: "modinfo br_netfilter" + environment: + PATH: "{{ ansible_env.PATH }}:/sbin" # Make sure we can workaround RH's conservative path management + register: modinfo_br_netfilter + failed_when: modinfo_br_netfilter.rc not in [0, 1] + changed_when: false + check_mode: no + +- name: Verify br_netfilter module path exists + file: + path: /etc/modules-load.d + state: directory + mode: 0755 + +- name: Enable br_netfilter module + community.general.modprobe: + name: br_netfilter + state: present + when: modinfo_br_netfilter.rc == 0 + +- name: Persist br_netfilter module + copy: + dest: /etc/modules-load.d/kubespray-br_netfilter.conf + content: br_netfilter + mode: 0644 + when: modinfo_br_netfilter.rc == 0 + +# kube-proxy needs net.bridge.bridge-nf-call-iptables enabled when found if br_netfilter is not a module +- name: Check if bridge-nf-call-iptables key exists + command: "sysctl net.bridge.bridge-nf-call-iptables" + failed_when: false + changed_when: false + check_mode: no + register: sysctl_bridge_nf_call_iptables + +- name: Enable bridge-nf-call tables + ansible.posix.sysctl: + name: "{{ item }}" + state: present + sysctl_file: "{{ sysctl_file_path }}" + value: "1" + reload: yes + when: sysctl_bridge_nf_call_iptables.rc == 0 + with_items: + - net.bridge.bridge-nf-call-iptables + - net.bridge.bridge-nf-call-arptables + - net.bridge.bridge-nf-call-ip6tables + +- name: Modprobe Kernel Module for IPVS + community.general.modprobe: + name: "{{ item }}" + state: present + with_items: + - ip_vs + - ip_vs_rr + - ip_vs_wrr + - ip_vs_sh + when: kube_proxy_mode == 'ipvs' + tags: + - kube-proxy + +- name: Modprobe nf_conntrack_ipv4 + community.general.modprobe: + name: nf_conntrack_ipv4 + state: present + register: modprobe_nf_conntrack_ipv4 + ignore_errors: true # noqa ignore-errors + when: + - kube_proxy_mode == 'ipvs' + tags: + - kube-proxy + +- name: Persist ip_vs modules + copy: + dest: /etc/modules-load.d/kube_proxy-ipvs.conf + mode: 0644 + content: | + ip_vs + ip_vs_rr + ip_vs_wrr + ip_vs_sh + {% if modprobe_nf_conntrack_ipv4 is success -%} + nf_conntrack_ipv4 + {%- endif -%} + when: kube_proxy_mode == 'ipvs' + tags: + - kube-proxy + +- name: Check cloud provider credentials + include_tasks: "cloud-credentials/{{ cloud_provider }}-credential-check.yml" + when: + - cloud_provider is defined + - cloud_provider in [ 'openstack', 'azure', 'vsphere' ] + tags: + - cloud-provider + - facts + +- name: Test if openstack_cacert is a base64 string + set_fact: + openstack_cacert_is_base64: "{% if openstack_cacert is search('^([A-Za-z0-9+/]{4})*([A-Za-z0-9+/]{3}= | [A-Za-z0-9+/]{2}==)?$') %}true{% else %}false{% endif %}" + when: + - cloud_provider is defined + - cloud_provider == 'openstack' + - openstack_cacert is defined + - openstack_cacert | length > 0 + + +- name: Write cacert file + copy: + src: "{{ openstack_cacert if not openstack_cacert_is_base64 else omit }}" + content: "{{ openstack_cacert | b64decode if openstack_cacert_is_base64 else omit }}" + dest: "{{ kube_config_dir }}/openstack-cacert.pem" + group: "{{ kube_cert_group }}" + mode: 0640 + when: + - cloud_provider is defined + - cloud_provider == 'openstack' + - openstack_cacert is defined + - openstack_cacert | length > 0 + tags: + - cloud-provider + +- name: Write cloud-config + template: + src: "cloud-configs/{{ cloud_provider }}-cloud-config.j2" + dest: "{{ kube_config_dir }}/cloud_config" + group: "{{ kube_cert_group }}" + mode: 0640 + when: + - cloud_provider is defined + - cloud_provider in [ 'openstack', 'azure', 'vsphere', 'aws', 'gce' ] + notify: Node | restart kubelet + tags: + - cloud-provider + +- name: Install kubelet + import_tasks: kubelet.yml + tags: + - kubelet + - kubeadm diff --git a/kubespray/roles/kubernetes/node/tasks/pre_upgrade.yml b/kubespray/roles/kubernetes/node/tasks/pre_upgrade.yml new file mode 100644 index 0000000..d9c2d07 --- /dev/null +++ b/kubespray/roles/kubernetes/node/tasks/pre_upgrade.yml @@ -0,0 +1,48 @@ +--- +- name: "Pre-upgrade | check if kubelet container exists" + shell: >- + set -o pipefail && + {% if container_manager in ['crio', 'docker'] %} + {{ docker_bin_dir }}/docker ps -af name=kubelet | grep kubelet + {% elif container_manager == 'containerd' %} + {{ bin_dir }}/crictl ps --all --name kubelet | grep kubelet + {% endif %} + args: + executable: /bin/bash + failed_when: false + changed_when: false + check_mode: no + register: kubelet_container_check + +- name: "Pre-upgrade | copy /var/lib/cni from kubelet" + command: >- + {% if container_manager in ['crio', 'docker'] %} + docker cp kubelet:/var/lib/cni /var/lib/cni + {% elif container_manager == 'containerd' %} + ctr run --rm --mount type=bind,src=/var/lib/cni,dst=/cnilibdir,options=rbind:rw kubelet kubelet-tmp sh -c 'cp /var/lib/cni/* /cnilibdir/' + {% endif %} + args: + creates: "/var/lib/cni" + failed_when: false + when: kubelet_container_check.rc == 0 + +- name: "Pre-upgrade | ensure kubelet container service is stopped if using host deployment" + service: + name: kubelet + state: stopped + when: kubelet_container_check.rc == 0 + +- name: "Pre-upgrade | ensure kubelet container is removed if using host deployment" + shell: >- + {% if container_manager in ['crio', 'docker'] %} + {{ docker_bin_dir }}/docker rm -fv kubelet + {% elif container_manager == 'containerd' %} + {{ bin_dir }}/crictl stop kubelet && {{ bin_dir }}/crictl rm kubelet + {% endif %} + failed_when: false + changed_when: false + register: remove_kubelet_container + retries: 4 + until: remove_kubelet_container.rc == 0 + delay: 5 + when: kubelet_container_check.rc == 0 diff --git a/kubespray/roles/kubernetes/node/templates/cloud-configs/aws-cloud-config.j2 b/kubespray/roles/kubernetes/node/templates/cloud-configs/aws-cloud-config.j2 new file mode 100644 index 0000000..f6d0c3d --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/cloud-configs/aws-cloud-config.j2 @@ -0,0 +1,11 @@ +[Global] +zone={{ aws_zone|default("") }} +vpc={{ aws_vpc|default("") }} +subnetId={{ aws_subnet_id|default("") }} +routeTableId={{ aws_route_table_id|default("") }} +roleArn={{ aws_role_arn|default("") }} +kubernetesClusterTag={{ aws_kubernetes_cluster_tag|default("") }} +kubernetesClusterId={{ aws_kubernetes_cluster_id|default("") }} +disableSecurityGroupIngress={{ "true" if aws_disable_security_group_ingress|default(False) else "false" }} +disableStrictZoneCheck={{ "true" if aws_disable_strict_zone_check|default(False) else "false" }} +elbSecurityGroup={{ aws_elb_security_group|default("") }} diff --git a/kubespray/roles/kubernetes/node/templates/cloud-configs/azure-cloud-config.j2 b/kubespray/roles/kubernetes/node/templates/cloud-configs/azure-cloud-config.j2 new file mode 100644 index 0000000..2b1c101 --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/cloud-configs/azure-cloud-config.j2 @@ -0,0 +1,26 @@ +{ + "cloud": "{{ azure_cloud }}", + "tenantId": "{{ azure_tenant_id }}", + "subscriptionId": "{{ azure_subscription_id }}", + "aadClientId": "{{ azure_aad_client_id }}", + "aadClientSecret": "{{ azure_aad_client_secret }}", + "resourceGroup": "{{ azure_resource_group }}", + "location": "{{ azure_location }}", + "subnetName": "{{ azure_subnet_name }}", + "securityGroupName": "{{ azure_security_group_name }}", + "securityGroupResourceGroup": "{{ azure_security_group_resource_group | default(azure_vnet_resource_group) }}", + "vnetName": "{{ azure_vnet_name }}", + "vnetResourceGroup": "{{ azure_vnet_resource_group }}", + "routeTableName": "{{ azure_route_table_name }}", + "routeTableResourceGroup": "{{ azure_route_table_resource_group | default(azure_vnet_resource_group) }}", + "vmType": "{{ azure_vmtype }}", +{% if azure_primary_availability_set_name is defined %} + "primaryAvailabilitySetName": "{{ azure_primary_availability_set_name }}", +{%endif%} + "useInstanceMetadata": {{azure_use_instance_metadata | lower }}, +{% if azure_loadbalancer_sku == "standard" %} + "excludeMasterFromStandardLB": {{ azure_exclude_master_from_standard_lb | lower }}, + "disableOutboundSNAT": {{ azure_disable_outbound_snat | lower }}, +{% endif%} + "loadBalancerSku": "{{ azure_loadbalancer_sku }}" +} diff --git a/kubespray/roles/kubernetes/node/templates/cloud-configs/gce-cloud-config.j2 b/kubespray/roles/kubernetes/node/templates/cloud-configs/gce-cloud-config.j2 new file mode 100644 index 0000000..3995126 --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/cloud-configs/gce-cloud-config.j2 @@ -0,0 +1,2 @@ +[global] +node-tags = {{ gce_node_tags }} diff --git a/kubespray/roles/kubernetes/node/templates/cloud-configs/openstack-cloud-config.j2 b/kubespray/roles/kubernetes/node/templates/cloud-configs/openstack-cloud-config.j2 new file mode 100644 index 0000000..b1f8e0a --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/cloud-configs/openstack-cloud-config.j2 @@ -0,0 +1,54 @@ +[Global] +auth-url="{{ openstack_auth_url }}" +username="{{ openstack_username }}" +password="{{ openstack_password }}" +region="{{ openstack_region }}" +{% if openstack_trust_id is defined and openstack_trust_id != "" %} +trust-id="{{ openstack_trust_id }}" +{% else %} +tenant-id="{{ openstack_tenant_id }}" +{% endif %} +{% if openstack_tenant_name is defined and openstack_tenant_name != "" %} +tenant-name="{{ openstack_tenant_name }}" +{% endif %} +{% if openstack_domain_name is defined and openstack_domain_name != "" %} +domain-name="{{ openstack_domain_name }}" +{% elif openstack_domain_id is defined and openstack_domain_id != "" %} +domain-id ="{{ openstack_domain_id }}" +{% endif %} +{% if openstack_cacert is defined and openstack_cacert != "" %} +ca-file="{{ kube_config_dir }}/openstack-cacert.pem" +{% endif %} + +[BlockStorage] +{% if openstack_blockstorage_version is defined %} +bs-version={{ openstack_blockstorage_version }} +{% endif %} +{% if openstack_blockstorage_ignore_volume_az is defined and openstack_blockstorage_ignore_volume_az|bool %} +ignore-volume-az={{ openstack_blockstorage_ignore_volume_az }} +{% endif %} +{% if node_volume_attach_limit is defined and node_volume_attach_limit != "" %} +node-volume-attach-limit="{{ node_volume_attach_limit }}" +{% endif %} + +{% if openstack_lbaas_enabled and openstack_lbaas_subnet_id is defined %} +[LoadBalancer] +subnet-id={{ openstack_lbaas_subnet_id }} +{% if openstack_lbaas_floating_network_id is defined %} +floating-network-id={{ openstack_lbaas_floating_network_id }} +{% endif %} +{% if openstack_lbaas_use_octavia is defined %} +use-octavia={{ openstack_lbaas_use_octavia }} +{% endif %} +{% if openstack_lbaas_method is defined %} +lb-method={{ openstack_lbaas_method }} +{% endif %} +{% if openstack_lbaas_provider is defined %} +lb-provider={{ openstack_lbaas_provider }} +{% endif %} + +create-monitor={{ openstack_lbaas_create_monitor }} +monitor-delay={{ openstack_lbaas_monitor_delay }} +monitor-timeout={{ openstack_lbaas_monitor_timeout }} +monitor-max-retries={{ openstack_lbaas_monitor_max_retries }} +{% endif %} diff --git a/kubespray/roles/kubernetes/node/templates/cloud-configs/vsphere-cloud-config.j2 b/kubespray/roles/kubernetes/node/templates/cloud-configs/vsphere-cloud-config.j2 new file mode 100644 index 0000000..2cda7f6 --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/cloud-configs/vsphere-cloud-config.j2 @@ -0,0 +1,36 @@ +[Global] +user = "{{ vsphere_user }}" +password = "{{ vsphere_password }}" +port = {{ vsphere_vcenter_port }} +insecure-flag = {{ vsphere_insecure }} + +datacenters = "{{ vsphere_datacenter }}" + +[VirtualCenter "{{ vsphere_vcenter_ip }}"] + + +[Workspace] +server = "{{ vsphere_vcenter_ip }}" +datacenter = "{{ vsphere_datacenter }}" +folder = "{{ vsphere_working_dir }}" +default-datastore = "{{ vsphere_datastore }}" +{% if vsphere_resource_pool is defined and vsphere_resource_pool != "" %} +resourcepool-path = "{{ vsphere_resource_pool }}" +{% endif %} + + +[Disk] +scsicontrollertype = {{ vsphere_scsi_controller_type }} + +{% if vsphere_public_network is defined and vsphere_public_network != "" %} +[Network] +public-network = {{ vsphere_public_network }} +{% endif %} + +[Labels] +{% if vsphere_zone_category is defined and vsphere_zone_category != "" %} +zone = {{ vsphere_zone_category }} +{% endif %} +{% if vsphere_region_category is defined and vsphere_region_category != "" %} +region = {{ vsphere_region_category }} +{% endif %} diff --git a/kubespray/roles/kubernetes/node/templates/http-proxy.conf.j2 b/kubespray/roles/kubernetes/node/templates/http-proxy.conf.j2 new file mode 100644 index 0000000..e790477 --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/http-proxy.conf.j2 @@ -0,0 +1,2 @@ +[Service] +Environment={% if http_proxy %}"HTTP_PROXY={{ http_proxy }}"{% endif %} {% if https_proxy %}"HTTPS_PROXY={{ https_proxy }}"{% endif %} {% if no_proxy %}"NO_PROXY={{ no_proxy }}"{% endif %} diff --git a/kubespray/roles/kubernetes/node/templates/kubelet-config.v1beta1.yaml.j2 b/kubespray/roles/kubernetes/node/templates/kubelet-config.v1beta1.yaml.j2 new file mode 100644 index 0000000..f54d1f8 --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/kubelet-config.v1beta1.yaml.j2 @@ -0,0 +1,170 @@ +apiVersion: kubelet.config.k8s.io/v1beta1 +kind: KubeletConfiguration +nodeStatusUpdateFrequency: "{{ kubelet_status_update_frequency }}" +failSwapOn: {{ kubelet_fail_swap_on }} +authentication: + anonymous: + enabled: false + webhook: + enabled: {{ kubelet_authentication_token_webhook }} + x509: + clientCAFile: {{ kube_cert_dir }}/ca.crt +authorization: +{% if kubelet_authorization_mode_webhook %} + mode: Webhook +{% else %} + mode: AlwaysAllow +{% endif %} +{% if kubelet_enforce_node_allocatable is defined and kubelet_enforce_node_allocatable != "\"\"" %} +{% set kubelet_enforce_node_allocatable_list = kubelet_enforce_node_allocatable.split(",") %} +enforceNodeAllocatable: +{% for item in kubelet_enforce_node_allocatable_list %} +- {{ item }} +{% endfor %} +{% endif %} +staticPodPath: {{ kube_manifest_dir }} +cgroupDriver: {{ kubelet_cgroup_driver | default('systemd') }} +containerLogMaxFiles: {{ kubelet_logfiles_max_nr }} +containerLogMaxSize: {{ kubelet_logfiles_max_size }} +maxPods: {{ kubelet_max_pods }} +podPidsLimit: {{ kubelet_pod_pids_limit }} +address: {{ kubelet_bind_address }} +readOnlyPort: {{ kube_read_only_port }} +healthzPort: {{ kubelet_healthz_port }} +healthzBindAddress: {{ kubelet_healthz_bind_address }} +kubeletCgroups: {{ kubelet_kubelet_cgroups }} +clusterDomain: {{ dns_domain }} +{% if kubelet_protect_kernel_defaults | bool %} +protectKernelDefaults: true +{% endif %} +{% if kubelet_rotate_certificates | bool %} +rotateCertificates: true +{% endif %} +{% if kubelet_rotate_server_certificates | bool %} +serverTLSBootstrap: true +{% endif %} +{# DNS settings for kubelet #} +{% if enable_nodelocaldns %} +{% set kubelet_cluster_dns = [nodelocaldns_ip] %} +{% elif dns_mode in ['coredns'] %} +{% set kubelet_cluster_dns = [skydns_server] %} +{% elif dns_mode == 'coredns_dual' %} +{% set kubelet_cluster_dns = [skydns_server,skydns_server_secondary] %} +{% elif dns_mode == 'manual' %} +{% set kubelet_cluster_dns = [manual_dns_server] %} +{% else %} +{% set kubelet_cluster_dns = [] %} +{% endif %} +clusterDNS: +{% for dns_address in kubelet_cluster_dns %} +- {{ dns_address }} +{% endfor %} +{# Node reserved CPU/memory #} +{% if kube_reserved | bool %} +kubeReservedCgroup: {{ kube_reserved_cgroups }} +kubeReserved: +{% if is_kube_master | bool %} + cpu: {{ kube_master_cpu_reserved }} + memory: {{ kube_master_memory_reserved }} +{% if kube_master_ephemeral_storage_reserved is defined %} + ephemeral-storage: {{ kube_master_ephemeral_storage_reserved }} +{% endif %} +{% if kube_master_pid_reserved is defined %} + pid: "{{ kube_master_pid_reserved }}" +{% endif %} +{% else %} + cpu: {{ kube_cpu_reserved }} + memory: {{ kube_memory_reserved }} +{% if kube_ephemeral_storage_reserved is defined %} + ephemeral-storage: {{ kube_ephemeral_storage_reserved }} +{% endif %} +{% if kube_pid_reserved is defined %} + pid: "{{ kube_pid_reserved }}" +{% endif %} +{% endif %} +{% endif %} +{% if system_reserved | bool %} +systemReservedCgroup: {{ system_reserved_cgroups }} +systemReserved: +{% if is_kube_master | bool %} + cpu: {{ system_master_cpu_reserved }} + memory: {{ system_master_memory_reserved }} +{% if system_master_ephemeral_storage_reserved is defined %} + ephemeral-storage: {{ system_master_ephemeral_storage_reserved }} +{% endif %} +{% if system_master_pid_reserved is defined %} + pid: "{{ system_master_pid_reserved }}" +{% endif %} +{% else %} + cpu: {{ system_cpu_reserved }} + memory: {{ system_memory_reserved }} +{% if system_ephemeral_storage_reserved is defined %} + ephemeral-storage: {{ system_ephemeral_storage_reserved }} +{% endif %} +{% if system_pid_reserved is defined %} + pid: "{{ system_pid_reserved }}" +{% endif %} +{% endif %} +{% endif %} +{% if is_kube_master | bool and eviction_hard_control_plane is defined and eviction_hard_control_plane %} +evictionHard: + {{ eviction_hard_control_plane | to_nice_yaml(indent=2) | indent(2) }} +{% elif not is_kube_master | bool and eviction_hard is defined and eviction_hard %} +evictionHard: + {{ eviction_hard | to_nice_yaml(indent=2) | indent(2) }} +{% endif %} +resolvConf: "{{ kube_resolv_conf }}" +{% if kubelet_config_extra_args %} +{{ kubelet_config_extra_args | to_nice_yaml(indent=2) }} +{% endif %} +{% if inventory_hostname in groups['kube_node'] and kubelet_node_config_extra_args %} +{{ kubelet_node_config_extra_args | to_nice_yaml(indent=2) }} +{% endif %} +{% if kubelet_feature_gates or kube_feature_gates %} +featureGates: +{% for feature in (kubelet_feature_gates | default(kube_feature_gates, true)) %} + {{ feature | replace("=", ": ") }} +{% endfor %} +{% endif %} +{% if tls_min_version is defined %} +tlsMinVersion: {{ tls_min_version }} +{% endif %} +{% if tls_cipher_suites is defined %} +tlsCipherSuites: +{% for tls in tls_cipher_suites %} +- {{ tls }} +{% endfor %} +{% endif %} +{% if kubelet_event_record_qps %} +eventRecordQPS: {{ kubelet_event_record_qps }} +{% endif %} +shutdownGracePeriod: {{ kubelet_shutdown_grace_period }} +shutdownGracePeriodCriticalPods: {{ kubelet_shutdown_grace_period_critical_pods }} +{% if not kubelet_fail_swap_on %} +memorySwap: + swapBehavior: {{ kubelet_swap_behavior }} +{% endif %} +{% if kubelet_streaming_connection_idle_timeout is defined %} +streamingConnectionIdleTimeout: {{ kubelet_streaming_connection_idle_timeout }} +{% endif %} +{% if kubelet_image_gc_high_threshold is defined %} +imageGCHighThresholdPercent: {{ kubelet_image_gc_high_threshold }} +{% endif %} +{% if kubelet_image_gc_low_threshold is defined %} +imageGCLowThresholdPercent: {{ kubelet_image_gc_low_threshold }} +{% endif %} +{% if kubelet_make_iptables_util_chains is defined %} +makeIPTablesUtilChains: {{ kubelet_make_iptables_util_chains | bool }} +{% endif %} +{% if kubelet_seccomp_default is defined %} +seccompDefault: {{ kubelet_seccomp_default | bool }} +{% endif %} +{% if kubelet_cpu_manager_policy is defined %} +cpuManagerPolicy: {{ kubelet_cpu_manager_policy }} +{% endif %} +{% if kubelet_topology_manager_policy is defined %} +topologyManagerPolicy: {{ kubelet_topology_manager_policy }} +{% endif %} +{% if kubelet_topology_manager_scope is defined %} +topologyManagerScope: {{ kubelet_topology_manager_scope }} +{% endif %} diff --git a/kubespray/roles/kubernetes/node/templates/kubelet.env.v1beta1.j2 b/kubespray/roles/kubernetes/node/templates/kubelet.env.v1beta1.j2 new file mode 100644 index 0000000..b8a22fd --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/kubelet.env.v1beta1.j2 @@ -0,0 +1,43 @@ +KUBE_LOG_LEVEL="--v={{ kube_log_level }}" +KUBELET_ADDRESS="--node-ip={{ kubelet_address }}" +{% if kube_override_hostname|default('') %} +KUBELET_HOSTNAME="--hostname-override={{ kube_override_hostname }}" +{% endif %} + +{# Base kubelet args #} +{% set kubelet_args_base -%} +{# start kubeadm specific settings #} +--bootstrap-kubeconfig=/etc/kubernetes/bootstrap-kubelet.conf \ +--config={{ kube_config_dir }}/kubelet-config.yaml \ +--kubeconfig={{ kube_config_dir }}/kubelet.conf \ +{# end kubeadm specific settings #} +--container-runtime-endpoint={{ cri_socket }} \ +--runtime-cgroups={{ kubelet_runtime_cgroups }} \ +{% endset %} + +{# Kubelet node taints for gpu #} +{% if nvidia_gpu_nodes is defined and nvidia_accelerator_enabled|bool %} +{% if inventory_hostname in nvidia_gpu_nodes and node_taints is defined %} +{% set dummy = node_taints.append('nvidia.com/gpu=:NoSchedule') %} +{% elif inventory_hostname in nvidia_gpu_nodes and node_taints is not defined %} +{% set node_taints = [] %} +{% set dummy = node_taints.append('nvidia.com/gpu=:NoSchedule') %} +{% endif %} +{% endif %} + +KUBELET_ARGS="{{ kubelet_args_base }} {% if node_taints|default([]) %}--register-with-taints={{ node_taints | join(',') }} {% endif %} {% if kubelet_custom_flags is string %} {{kubelet_custom_flags}} {% else %}{% for flag in kubelet_custom_flags %} {{flag}} {% endfor %}{% endif %}{% if inventory_hostname in groups['kube_node'] %}{% if kubelet_node_custom_flags is string %} {{kubelet_node_custom_flags}} {% else %}{% for flag in kubelet_node_custom_flags %} {{flag}} {% endfor %}{% endif %}{% endif %}" +{% if kubelet_flexvolumes_plugins_dir is defined %} +KUBELET_VOLUME_PLUGIN="--volume-plugin-dir={{ kubelet_flexvolumes_plugins_dir }}" +{% endif %} +{% if kube_network_plugin is defined and kube_network_plugin == "cloud" %} +KUBELET_NETWORK_PLUGIN="--hairpin-mode=promiscuous-bridge --network-plugin=kubenet" +{% endif %} +{% if cloud_provider is defined and cloud_provider in ["openstack", "azure", "vsphere", "aws", "gce"] %} +KUBELET_CLOUDPROVIDER="--cloud-provider={{ cloud_provider }} --cloud-config={{ kube_config_dir }}/cloud_config" +{% elif cloud_provider is defined and cloud_provider in ["external"] %} +KUBELET_CLOUDPROVIDER="--cloud-provider={{ cloud_provider }}" +{% else %} +KUBELET_CLOUDPROVIDER="" +{% endif %} + +PATH={{ bin_dir }}:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin diff --git a/kubespray/roles/kubernetes/node/templates/kubelet.service.j2 b/kubespray/roles/kubernetes/node/templates/kubelet.service.j2 new file mode 100644 index 0000000..9df98e0 --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/kubelet.service.j2 @@ -0,0 +1,52 @@ +[Unit] +Description=Kubernetes Kubelet Server +Documentation=https://github.com/GoogleCloudPlatform/kubernetes +After={{ container_manager }}.service +{% if container_manager == 'docker' %} +Wants=docker.socket +{% else %} +Wants={{ container_manager }}.service +{% endif %} + +[Service] +EnvironmentFile=-{{ kube_config_dir }}/kubelet.env +{% if system_reserved|bool %} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/cpu/{{ system_reserved_cgroups_for_service_slice }} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/cpuacct/{{ system_reserved_cgroups_for_service_slice }} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/cpuset/{{ system_reserved_cgroups_for_service_slice }} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/hugetlb/{{ system_reserved_cgroups_for_service_slice }} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/memory/{{ system_reserved_cgroups_for_service_slice }} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/pids/{{ system_reserved_cgroups_for_service_slice }} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/systemd/{{ system_reserved_cgroups_for_service_slice }} +{% endif %} +{% if kube_reserved|bool %} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/cpu/{{ kube_reserved_cgroups_for_service_slice }} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/cpuacct/{{ kube_reserved_cgroups_for_service_slice }} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/cpuset/{{ kube_reserved_cgroups_for_service_slice }} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/hugetlb/{{ kube_reserved_cgroups_for_service_slice }} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/memory/{{ kube_reserved_cgroups_for_service_slice }} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/pids/{{ kube_reserved_cgroups_for_service_slice }} +ExecStartPre=/bin/mkdir -p /sys/fs/cgroup/systemd/{{ kube_reserved_cgroups_for_service_slice }} +{% endif %} +ExecStart={{ bin_dir }}/kubelet \ + $KUBE_LOGTOSTDERR \ + $KUBE_LOG_LEVEL \ + $KUBELET_API_SERVER \ + $KUBELET_ADDRESS \ + $KUBELET_PORT \ + $KUBELET_HOSTNAME \ + $KUBELET_ARGS \ + $DOCKER_SOCKET \ + $KUBELET_NETWORK_PLUGIN \ + $KUBELET_VOLUME_PLUGIN \ + $KUBELET_CLOUDPROVIDER +Restart=always +RestartSec=10s +{% if kubelet_systemd_hardening %} +# Hardening setup +IPAddressDeny=any +IPAddressAllow={{ kubelet_secure_addresses }} +{% endif %} + +[Install] +WantedBy=multi-user.target diff --git a/kubespray/roles/kubernetes/node/templates/loadbalancer/haproxy.cfg.j2 b/kubespray/roles/kubernetes/node/templates/loadbalancer/haproxy.cfg.j2 new file mode 100644 index 0000000..c629325 --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/loadbalancer/haproxy.cfg.j2 @@ -0,0 +1,49 @@ +global + maxconn 4000 + log 127.0.0.1 local0 + +defaults + mode http + log global + option httplog + option dontlognull + option http-server-close + option redispatch + retries 5 + timeout http-request 5m + timeout queue 5m + timeout connect 30s + timeout client {{ loadbalancer_apiserver_keepalive_timeout }} + timeout server 15m + timeout http-keep-alive 30s + timeout check 30s + maxconn 4000 + +{% if loadbalancer_apiserver_healthcheck_port is defined -%} +frontend healthz + bind 0.0.0.0:{{ loadbalancer_apiserver_healthcheck_port }} + {% if enable_dual_stack_networks -%} + bind :::{{ loadbalancer_apiserver_healthcheck_port }} + {% endif -%} + mode http + monitor-uri /healthz +{% endif %} + +frontend kube_api_frontend + bind 127.0.0.1:{{ loadbalancer_apiserver_port|default(kube_apiserver_port) }} + {% if enable_dual_stack_networks -%} + bind [::1]:{{ loadbalancer_apiserver_port|default(kube_apiserver_port) }}; + {% endif -%} + mode tcp + option tcplog + default_backend kube_api_backend + +backend kube_api_backend + mode tcp + balance leastconn + default-server inter 15s downinter 15s rise 2 fall 2 slowstart 60s maxconn 1000 maxqueue 256 weight 100 + option httpchk GET /healthz + http-check expect status 200 + {% for host in groups['kube_control_plane'] -%} + server {{ host }} {{ hostvars[host]['access_ip'] | default(hostvars[host]['ip'] | default(fallback_ips[host])) }}:{{ kube_apiserver_port }} check check-ssl verify none + {% endfor -%} diff --git a/kubespray/roles/kubernetes/node/templates/loadbalancer/nginx.conf.j2 b/kubespray/roles/kubernetes/node/templates/loadbalancer/nginx.conf.j2 new file mode 100644 index 0000000..07b9370 --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/loadbalancer/nginx.conf.j2 @@ -0,0 +1,60 @@ +error_log stderr notice; + +worker_processes 2; +worker_rlimit_nofile 130048; +worker_shutdown_timeout 10s; + +events { + multi_accept on; + use epoll; + worker_connections 16384; +} + +stream { + upstream kube_apiserver { + least_conn; + {% for host in groups['kube_control_plane'] -%} + server {{ hostvars[host]['access_ip'] | default(hostvars[host]['ip'] | default(fallback_ips[host])) }}:{{ kube_apiserver_port }}; + {% endfor -%} + } + + server { + listen 127.0.0.1:{{ loadbalancer_apiserver_port|default(kube_apiserver_port) }}; + {% if enable_dual_stack_networks -%} + listen [::1]:{{ loadbalancer_apiserver_port|default(kube_apiserver_port) }}; + {% endif -%} + proxy_pass kube_apiserver; + proxy_timeout 10m; + proxy_connect_timeout 1s; + } +} + +http { + aio threads; + aio_write on; + tcp_nopush on; + tcp_nodelay on; + + keepalive_timeout {{ loadbalancer_apiserver_keepalive_timeout }}; + keepalive_requests 100; + reset_timedout_connection on; + server_tokens off; + autoindex off; + + {% if loadbalancer_apiserver_healthcheck_port is defined -%} + server { + listen {{ loadbalancer_apiserver_healthcheck_port }}; + {% if enable_dual_stack_networks -%} + listen [::]:{{ loadbalancer_apiserver_healthcheck_port }}; + {% endif -%} + location /healthz { + access_log off; + return 200; + } + location /stub_status { + stub_status on; + access_log off; + } + } + {% endif %} +} diff --git a/kubespray/roles/kubernetes/node/templates/manifests/haproxy.manifest.j2 b/kubespray/roles/kubernetes/node/templates/manifests/haproxy.manifest.j2 new file mode 100644 index 0000000..7c5097c --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/manifests/haproxy.manifest.j2 @@ -0,0 +1,42 @@ +apiVersion: v1 +kind: Pod +metadata: + name: {{ loadbalancer_apiserver_pod_name }} + namespace: kube-system + labels: + addonmanager.kubernetes.io/mode: Reconcile + k8s-app: kube-haproxy + annotations: + haproxy-cfg-checksum: "{{ haproxy_stat.stat.checksum }}" +spec: + hostNetwork: true + dnsPolicy: ClusterFirstWithHostNet + nodeSelector: + kubernetes.io/os: linux + priorityClassName: system-node-critical + containers: + - name: haproxy + image: {{ haproxy_image_repo }}:{{ haproxy_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + resources: + requests: + cpu: {{ loadbalancer_apiserver_cpu_requests }} + memory: {{ loadbalancer_apiserver_memory_requests }} + {% if loadbalancer_apiserver_healthcheck_port is defined -%} + livenessProbe: + httpGet: + path: /healthz + port: {{ loadbalancer_apiserver_healthcheck_port }} + readinessProbe: + httpGet: + path: /healthz + port: {{ loadbalancer_apiserver_healthcheck_port }} + {% endif -%} + volumeMounts: + - mountPath: /usr/local/etc/haproxy/ + name: etc-haproxy + readOnly: true + volumes: + - name: etc-haproxy + hostPath: + path: {{ haproxy_config_dir }} diff --git a/kubespray/roles/kubernetes/node/templates/manifests/kube-vip.manifest.j2 b/kubespray/roles/kubernetes/node/templates/manifests/kube-vip.manifest.j2 new file mode 100644 index 0000000..b95b474 --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/manifests/kube-vip.manifest.j2 @@ -0,0 +1,102 @@ +# Inspired by https://github.com/kube-vip/kube-vip/blob/v0.5.11/pkg/kubevip/config_generator.go#L13 +apiVersion: v1 +kind: Pod +metadata: + name: kube-vip + namespace: kube-system + labels: + addonmanager.kubernetes.io/mode: Reconcile + k8s-app: kube-vip +spec: + containers: + - args: + - manager + env: + - name: vip_arp + value: {{ kube_vip_arp_enabled | string | to_json }} + - name: port + value: {{ kube_apiserver_port | string | to_json }} +{% if kube_vip_interface %} + - name: vip_interface + value: {{ kube_vip_interface | string | to_json }} +{% endif %} +{% if kube_vip_services_interface %} + - name: vip_servicesinterface + value: {{ kube_vip_services_interface | string | to_json }} +{% endif %} +{% if kube_vip_cidr %} + - name: vip_cidr + value: {{ kube_vip_cidr | string | to_json }} +{% endif %} +{% if kube_vip_controlplane_enabled %} + - name: cp_enable + value: "true" + - name: cp_namespace + value: kube-system + - name: vip_ddns + value: {{ kube_vip_ddns_enabled | string | to_json }} +{% endif %} +{% if kube_vip_services_enabled %} + - name: svc_enable + value: "true" +{% endif %} +{% if kube_vip_enableServicesElection %} + - name: svc_election + value: "true" +{% endif %} +{% if kube_vip_leader_election_enabled %} + - name: vip_leaderelection + value: "true" + - name: vip_leaseduration + value: "5" + - name: vip_renewdeadline + value: "3" + - name: vip_retryperiod + value: "1" +{% endif %} +{% if kube_vip_bgp_enabled %} + - name: bgp_enable + value: "true" + - name: bgp_routerid + value: {{ kube_vip_bgp_routerid | string | to_json }} + - name: bgp_as + value: {{ kube_vip_local_as | string | to_json }} + - name: bgp_peeraddress + value: {{ kube_vip_bgp_peeraddress | to_json }} + - name: bgp_peerpass + value: {{ kube_vip_bgp_peerpass | to_json }} + - name: bgp_peeras + value: {{ kube_vip_bgp_peeras | string | to_json }} +{% if kube_vip_bgppeers %} + - name: bgp_peers + value: {{ kube_vip_bgppeers | join(',') | to_json }} +{% endif %} +{% endif %} + - name: address + value: {{ kube_vip_address | to_json }} +{% if kube_vip_lb_enable %} + - name: lb_enable + value: "true" +{% endif %} + image: {{ kube_vip_image_repo }}:{{ kube_vip_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + name: kube-vip + resources: {} + securityContext: + capabilities: + add: + - NET_ADMIN + - NET_RAW + volumeMounts: + - mountPath: /etc/kubernetes/admin.conf + name: kubeconfig + hostAliases: + - hostnames: + - kubernetes + ip: 127.0.0.1 + hostNetwork: true + volumes: + - hostPath: + path: /etc/kubernetes/admin.conf + name: kubeconfig +status: {} diff --git a/kubespray/roles/kubernetes/node/templates/manifests/nginx-proxy.manifest.j2 b/kubespray/roles/kubernetes/node/templates/manifests/nginx-proxy.manifest.j2 new file mode 100644 index 0000000..16757ec --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/manifests/nginx-proxy.manifest.j2 @@ -0,0 +1,42 @@ +apiVersion: v1 +kind: Pod +metadata: + name: {{ loadbalancer_apiserver_pod_name }} + namespace: kube-system + labels: + addonmanager.kubernetes.io/mode: Reconcile + k8s-app: kube-nginx + annotations: + nginx-cfg-checksum: "{{ nginx_stat.stat.checksum }}" +spec: + hostNetwork: true + dnsPolicy: ClusterFirstWithHostNet + nodeSelector: + kubernetes.io/os: linux + priorityClassName: system-node-critical + containers: + - name: nginx-proxy + image: {{ nginx_image_repo }}:{{ nginx_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + resources: + requests: + cpu: {{ loadbalancer_apiserver_cpu_requests }} + memory: {{ loadbalancer_apiserver_memory_requests }} + {% if loadbalancer_apiserver_healthcheck_port is defined -%} + livenessProbe: + httpGet: + path: /healthz + port: {{ loadbalancer_apiserver_healthcheck_port }} + readinessProbe: + httpGet: + path: /healthz + port: {{ loadbalancer_apiserver_healthcheck_port }} + {% endif -%} + volumeMounts: + - mountPath: /etc/nginx + name: etc-nginx + readOnly: true + volumes: + - name: etc-nginx + hostPath: + path: {{ nginx_config_dir }} diff --git a/kubespray/roles/kubernetes/node/templates/node-kubeconfig.yaml.j2 b/kubespray/roles/kubernetes/node/templates/node-kubeconfig.yaml.j2 new file mode 100644 index 0000000..4b8af60 --- /dev/null +++ b/kubespray/roles/kubernetes/node/templates/node-kubeconfig.yaml.j2 @@ -0,0 +1,19 @@ +--- +apiVersion: v1 +kind: Config +clusters: +- name: local + cluster: + certificate-authority: {{ kube_cert_dir }}/ca.pem + server: {{ kube_apiserver_endpoint }} +users: +- name: kubelet + user: + client-certificate: {{ kube_cert_dir }}/node-{{ inventory_hostname }}.pem + client-key: {{ kube_cert_dir }}/node-{{ inventory_hostname }}-key.pem +contexts: +- context: + cluster: local + user: kubelet + name: kubelet-{{ cluster_name }} +current-context: kubelet-{{ cluster_name }} diff --git a/kubespray/roles/kubernetes/node/vars/fedora.yml b/kubespray/roles/kubernetes/node/vars/fedora.yml new file mode 100644 index 0000000..59bc55d --- /dev/null +++ b/kubespray/roles/kubernetes/node/vars/fedora.yml @@ -0,0 +1,2 @@ +--- +kube_resolv_conf: "/run/systemd/resolve/resolv.conf" diff --git a/kubespray/roles/kubernetes/node/vars/ubuntu-18.yml b/kubespray/roles/kubernetes/node/vars/ubuntu-18.yml new file mode 100644 index 0000000..59bc55d --- /dev/null +++ b/kubespray/roles/kubernetes/node/vars/ubuntu-18.yml @@ -0,0 +1,2 @@ +--- +kube_resolv_conf: "/run/systemd/resolve/resolv.conf" diff --git a/kubespray/roles/kubernetes/node/vars/ubuntu-20.yml b/kubespray/roles/kubernetes/node/vars/ubuntu-20.yml new file mode 100644 index 0000000..59bc55d --- /dev/null +++ b/kubespray/roles/kubernetes/node/vars/ubuntu-20.yml @@ -0,0 +1,2 @@ +--- +kube_resolv_conf: "/run/systemd/resolve/resolv.conf" diff --git a/kubespray/roles/kubernetes/node/vars/ubuntu-22.yml b/kubespray/roles/kubernetes/node/vars/ubuntu-22.yml new file mode 100644 index 0000000..59bc55d --- /dev/null +++ b/kubespray/roles/kubernetes/node/vars/ubuntu-22.yml @@ -0,0 +1,2 @@ +--- +kube_resolv_conf: "/run/systemd/resolve/resolv.conf" diff --git a/kubespray/roles/kubernetes/preinstall/defaults/main.yml b/kubespray/roles/kubernetes/preinstall/defaults/main.yml new file mode 100644 index 0000000..f767031 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/defaults/main.yml @@ -0,0 +1,152 @@ +--- +# Set to true to allow pre-checks to fail and continue deployment +ignore_assert_errors: false + +epel_enabled: false +# Kubespray sets this to true after clusterDNS is running to apply changes to the host resolv.conf +dns_late: false + +common_required_pkgs: + - "{{ (ansible_distribution == 'openSUSE Tumbleweed') | ternary('openssl-1_1', 'openssl') }}" + - curl + - rsync + - socat + - unzip + - e2fsprogs + - xfsprogs + - ebtables + - bash-completion + - tar + +# Set to true if your network does not support IPv6 +# This maybe necessary for pulling Docker images from +# GCE docker repository +disable_ipv6_dns: false + +kube_owner: kube +kube_cert_group: kube-cert +kube_config_dir: /etc/kubernetes +kube_cert_dir: "{{ kube_config_dir }}/ssl" +kube_cert_compat_dir: /etc/kubernetes/pki +kubelet_flexvolumes_plugins_dir: /usr/libexec/kubernetes/kubelet-plugins/volume/exec + +# Flatcar Container Linux by Kinvolk cloud init config file to define /etc/resolv.conf content +# for hostnet pods and infra needs +resolveconf_cloud_init_conf: /etc/resolveconf_cloud_init.conf + +# All inventory hostnames will be written into each /etc/hosts file. +populate_inventory_to_hosts_file: true +# K8S Api FQDN will be written into /etc/hosts file. +populate_loadbalancer_apiserver_to_hosts_file: true +# etc_hosts_localhost_entries will be written into /etc/hosts file. +populate_localhost_entries_to_hosts_file: true + +sysctl_file_path: "/etc/sysctl.d/99-sysctl.conf" + +etc_hosts_localhost_entries: + 127.0.0.1: + expected: + - localhost + - localhost.localdomain + ::1: + expected: + - localhost6 + - localhost6.localdomain + unexpected: + - localhost + - localhost.localdomain + +# Minimal memory requirement in MB for safety checks +minimal_node_memory_mb: 1024 +minimal_master_memory_mb: 1500 + +yum_repo_dir: /etc/yum.repos.d + +# number of times package install task should be retried +pkg_install_retries: 4 + +# Check if access_ip responds to ping. Set false if your firewall blocks ICMP. +ping_access_ip: true + +## NTP Settings +# Start the ntpd or chrony service and enable it at system boot. +ntp_enabled: false +# The package to install which provides NTP functionality. +# The default is ntp for most platforms, or chrony on RHEL/CentOS 7 and later. +# The ntp_package can be one of ['ntp', 'chrony'] +ntp_package: >- + {% if ansible_os_family == "RedHat" -%} + chrony + {%- else -%} + ntp + {%- endif -%} + +# Manage the NTP configuration file. +ntp_manage_config: false +# Specify the NTP servers +# Only takes effect when ntp_manage_config is true. +ntp_servers: + - "0.pool.ntp.org iburst" + - "1.pool.ntp.org iburst" + - "2.pool.ntp.org iburst" + - "3.pool.ntp.org iburst" +# Restrict NTP access to these hosts. +# Only takes effect when ntp_manage_config is true. +ntp_restrict: + - "127.0.0.1" + - "::1" +# The NTP driftfile path +# Only takes effect when ntp_manage_config is true. +ntp_driftfile: /var/lib/ntp/ntp.drift +# Enable tinker panic is useful when running NTP in a VM environment. +# Only takes effect when ntp_manage_config is true. +ntp_tinker_panic: false + +# Force sync time immediately after the ntp installed, which is useful in in newly installed system. +ntp_force_sync_immediately: false + +# Set the timezone for your server. eg: "Etc/UTC","Etc/GMT-8". If not set, the timezone will not change. +ntp_timezone: "" + +# Currently known os distributions +supported_os_distributions: + - 'RedHat' + - 'CentOS' + - 'Fedora' + - 'Ubuntu' + - 'Debian' + - 'Flatcar' + - 'Flatcar Container Linux by Kinvolk' + - 'Suse' + - 'openSUSE Leap' + - 'openSUSE Tumbleweed' + - 'ClearLinux' + - 'OracleLinux' + - 'AlmaLinux' + - 'Rocky' + - 'Amazon' + - 'Kylin Linux Advanced Server' + - 'UnionTech' + - 'UniontechOS' + - 'openEuler' + +# Extending some distributions into the redhat os family +redhat_os_family_extensions: + - "Kylin Linux Advanced Server" + - "openEuler" + - "UnionTech" + - "UniontechOS" + +# Extending some distributions into the debian os family +debian_os_family_extensions: + - "UnionTech OS Server 20" + +# Sets DNSStubListener=no, useful if you get "0.0.0.0:53: bind: address already in use" +systemd_resolved_disable_stub_listener: "{{ ansible_os_family in ['Flatcar', 'Flatcar Container Linux by Kinvolk'] }}" + +# Used to disable File Access Policy Daemon service. +# If service is enabled, the CNI plugin installation will fail +disable_fapolicyd: true + +# Enable 0120-growpart-azure-centos-7 tasks +growpart_azure_enabled: true diff --git a/kubespray/roles/kubernetes/preinstall/files/dhclient_nodnsupdate b/kubespray/roles/kubernetes/preinstall/files/dhclient_nodnsupdate new file mode 100644 index 0000000..03c7c99 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/files/dhclient_nodnsupdate @@ -0,0 +1,4 @@ +#!/bin/sh +make_resolv_conf() { + : +} diff --git a/kubespray/roles/kubernetes/preinstall/gen-gitinfos.sh b/kubespray/roles/kubernetes/preinstall/gen-gitinfos.sh new file mode 100755 index 0000000..bfab5a4 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/gen-gitinfos.sh @@ -0,0 +1,73 @@ +#!/bin/sh +set -e + +# Text color variables +txtbld=$(tput bold) # Bold +bldred=${txtbld}$(tput setaf 1) # red +bldgre=${txtbld}$(tput setaf 2) # green +bldylw=${txtbld}$(tput setaf 3) # yellow +txtrst=$(tput sgr0) # Reset +err=${bldred}ERROR${txtrst} +info=${bldgre}INFO${txtrst} +warn=${bldylw}WARNING${txtrst} + +usage() +{ + cat << EOF +Generates a file which contains useful git informations + +Usage : $(basename $0) [global|diff] + ex : + Generate git information + $(basename $0) global + Generate diff from latest tag + $(basename $0) diff +EOF +} + +if [ $# != 1 ]; then + printf "\n$err : Needs 1 argument\n" + usage + exit 2 +fi; + +current_commit=$(git rev-parse HEAD) +latest_tag=$(git describe --abbrev=0 --tags) +latest_tag_commit=$(git show-ref -s ${latest_tag}) +tags_list=$(git tag --points-at "${latest_tag}") + +case ${1} in + "global") +cat<0 + fail_msg: "nameserver should not empty in /etc/resolv.conf" + when: + - not ignore_assert_errors + - configured_nameservers is defined + - not (upstream_dns_servers is defined and upstream_dns_servers | length > 0) + - not (disable_host_nameservers | default(false)) + +- name: NetworkManager | Check if host has NetworkManager + # noqa command-instead-of-module - Should we use service_facts for this? + command: systemctl is-active --quiet NetworkManager.service + register: networkmanager_enabled + failed_when: false + changed_when: false + check_mode: false + +- name: Check systemd-resolved + # noqa command-instead-of-module - Should we use service_facts for this? + command: systemctl is-active systemd-resolved + register: systemd_resolved_enabled + failed_when: false + changed_when: false + check_mode: no + +- name: Set default dns if remove_default_searchdomains is false + set_fact: + default_searchdomains: ["default.svc.{{ dns_domain }}", "svc.{{ dns_domain }}"] + when: not remove_default_searchdomains | default() | bool or (remove_default_searchdomains | default() | bool and searchdomains | default([]) | length==0) + +- name: Set dns facts + set_fact: + resolvconf: >- + {%- if resolvconf.rc == 0 and resolvconfd_path.stat.isdir is defined and resolvconfd_path.stat.isdir -%}true{%- else -%}false{%- endif -%} + bogus_domains: |- + {% for d in default_searchdomains | default([]) + searchdomains | default([]) -%} + {{ dns_domain }}.{{ d }}./{{ d }}.{{ d }}./com.{{ d }}./ + {%- endfor %} + cloud_resolver: "{{ ['169.254.169.254'] if cloud_provider is defined and cloud_provider == 'gce' else + ['169.254.169.253'] if cloud_provider is defined and cloud_provider == 'aws' else + [] }}" + +- name: Check if kubelet is configured + stat: + path: "{{ kube_config_dir }}/kubelet.env" + get_attributes: no + get_checksum: no + get_mime: no + register: kubelet_configured + changed_when: false + +- name: Check if early DNS configuration stage + set_fact: + dns_early: "{{ not kubelet_configured.stat.exists }}" + +- name: Target resolv.conf files + set_fact: + resolvconffile: /etc/resolv.conf + base: >- + {%- if resolvconf | bool -%}/etc/resolvconf/resolv.conf.d/base{%- endif -%} + head: >- + {%- if resolvconf | bool -%}/etc/resolvconf/resolv.conf.d/head{%- endif -%} + when: not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] and not is_fedora_coreos + +- name: Target temporary resolvconf cloud init file (Flatcar Container Linux by Kinvolk / Fedora CoreOS) + set_fact: + resolvconffile: /tmp/resolveconf_cloud_init_conf + when: ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] or is_fedora_coreos + +- name: Check if /etc/dhclient.conf exists + stat: + path: /etc/dhclient.conf + get_attributes: no + get_checksum: no + get_mime: no + register: dhclient_stat + +- name: Target dhclient conf file for /etc/dhclient.conf + set_fact: + dhclientconffile: /etc/dhclient.conf + when: dhclient_stat.stat.exists + +- name: Check if /etc/dhcp/dhclient.conf exists + stat: + path: /etc/dhcp/dhclient.conf + get_attributes: no + get_checksum: no + get_mime: no + register: dhcp_dhclient_stat + +- name: Target dhclient conf file for /etc/dhcp/dhclient.conf + set_fact: + dhclientconffile: /etc/dhcp/dhclient.conf + when: dhcp_dhclient_stat.stat.exists + +- name: Target dhclient hook file for Red Hat family + set_fact: + dhclienthookfile: /etc/dhcp/dhclient.d/zdnsupdate.sh + when: ansible_os_family == "RedHat" + +- name: Target dhclient hook file for Debian family + set_fact: + dhclienthookfile: /etc/dhcp/dhclient-exit-hooks.d/zdnsupdate + when: ansible_os_family == "Debian" + +- name: Generate search domains to resolvconf + set_fact: + searchentries: + search {{ (default_searchdomains | default([]) + searchdomains | default([])) | join(' ') }} + domainentry: + domain {{ dns_domain }} + supersede_search: + supersede domain-search "{{ (default_searchdomains | default([]) + searchdomains | default([])) | join('", "') }}"; + supersede_domain: + supersede domain-name "{{ dns_domain }}"; + +- name: Pick coredns cluster IP or default resolver + set_fact: + coredns_server: |- + {%- if dns_mode == 'coredns' and not dns_early | bool -%} + {{ [skydns_server] }} + {%- elif dns_mode == 'coredns_dual' and not dns_early | bool -%} + {{ [skydns_server] + [skydns_server_secondary] }} + {%- elif dns_mode == 'manual' and not dns_early | bool -%} + {{ (manual_dns_server.split(',') | list) }} + {%- elif dns_mode == 'none' and not dns_early | bool -%} + [] + {%- elif dns_early | bool -%} + {{ upstream_dns_servers | default([]) }} + {%- endif -%} + +# This task should only run after cluster/nodelocal DNS is up, otherwise all DNS lookups will timeout +- name: Generate nameservers for resolvconf, including cluster DNS + set_fact: + nameserverentries: |- + {{ (([nodelocaldns_ip] if enable_nodelocaldns else []) + (coredns_server | d([]) if not enable_nodelocaldns else []) + nameservers | d([]) + cloud_resolver | d([]) + (configured_nameservers | d([]) if not disable_host_nameservers | d() | bool else [])) | unique | join(',') }} + supersede_nameserver: + supersede domain-name-servers {{ (([nodelocaldns_ip] if enable_nodelocaldns else []) + (coredns_server | d([]) if not enable_nodelocaldns else []) + nameservers | d([]) + cloud_resolver | d([])) | unique | join(', ') }}; + when: not dns_early or dns_late + +# This task should run instead of the above task when cluster/nodelocal DNS hasn't +# been deployed yet (like scale.yml/cluster.yml) or when it's down (reset.yml) +- name: Generate nameservers for resolvconf, not including cluster DNS + set_fact: + nameserverentries: |- + {{ (nameservers | d([]) + cloud_resolver | d([]) + configured_nameservers | d([])) | unique | join(',') }} + supersede_nameserver: + supersede domain-name-servers {{ (nameservers | d([]) + cloud_resolver | d([])) | unique | join(', ') }}; + when: dns_early and not dns_late + +- name: Gather os specific variables + include_vars: "{{ item }}" + with_first_found: + - files: + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_release }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_major_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}.yml" + - "{{ ansible_os_family | lower }}.yml" + - defaults.yml + paths: + - ../vars + skip: true + +- name: Set etcd vars if using kubeadm mode + set_fact: + etcd_cert_dir: "{{ kube_cert_dir }}" + kube_etcd_cacert_file: "etcd/ca.crt" + kube_etcd_cert_file: "apiserver-etcd-client.crt" + kube_etcd_key_file: "apiserver-etcd-client.key" + when: + - etcd_deployment_type == "kubeadm" + +- name: Check /usr readonly + stat: + path: "/usr" + get_attributes: no + get_checksum: no + get_mime: no + register: usr + +- name: Set alternate flexvolume path + set_fact: + kubelet_flexvolumes_plugins_dir: /var/lib/kubelet/volumeplugins + when: not usr.stat.writeable diff --git a/kubespray/roles/kubernetes/preinstall/tasks/0040-verify-settings.yml b/kubespray/roles/kubernetes/preinstall/tasks/0040-verify-settings.yml new file mode 100644 index 0000000..8cc11b6 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/0040-verify-settings.yml @@ -0,0 +1,318 @@ +--- +- name: Stop if either kube_control_plane or kube_node group is empty + assert: + that: "groups.get('{{ item }}')" + with_items: + - kube_control_plane + - kube_node + run_once: true + when: not ignore_assert_errors + +- name: Stop if etcd group is empty in external etcd mode + assert: + that: groups.get('etcd') + fail_msg: "Group 'etcd' cannot be empty in external etcd mode" + run_once: true + when: + - not ignore_assert_errors + - etcd_deployment_type != "kubeadm" + +- name: Stop if non systemd OS type + assert: + that: ansible_service_mgr == "systemd" + when: not ignore_assert_errors + +- name: Stop if the os does not support + assert: + that: (allow_unsupported_distribution_setup | default(false)) or ansible_distribution in supported_os_distributions + msg: "{{ ansible_distribution }} is not a known OS" + when: not ignore_assert_errors + +- name: Stop if unknown network plugin + assert: + that: kube_network_plugin in ['calico', 'flannel', 'weave', 'cloud', 'cilium', 'cni', 'kube-ovn', 'kube-router', 'macvlan', 'custom_cni'] + msg: "{{ kube_network_plugin }} is not supported" + when: + - kube_network_plugin is defined + - not ignore_assert_errors + +- name: Stop if unsupported version of Kubernetes + assert: + that: kube_version is version(kube_version_min_required, '>=') + msg: "The current release of Kubespray only support newer version of Kubernetes than {{ kube_version_min_required }} - You are trying to apply {{ kube_version }}" + when: not ignore_assert_errors + +# simplify this items-list when https://github.com/ansible/ansible/issues/15753 is resolved +- name: "Stop if known booleans are set as strings (Use JSON format on CLI: -e \"{'key': true }\")" + assert: + that: item.value | type_debug == 'bool' + msg: "{{ item.value }} isn't a bool" + run_once: yes + with_items: + - { name: download_run_once, value: "{{ download_run_once }}" } + - { name: deploy_netchecker, value: "{{ deploy_netchecker }}" } + - { name: download_always_pull, value: "{{ download_always_pull }}" } + - { name: helm_enabled, value: "{{ helm_enabled }}" } + - { name: openstack_lbaas_enabled, value: "{{ openstack_lbaas_enabled }}" } + when: not ignore_assert_errors + +- name: Stop if even number of etcd hosts + assert: + that: groups.etcd | length is not divisibleby 2 + when: + - not ignore_assert_errors + - inventory_hostname in groups.get('etcd',[]) + +- name: Stop if memory is too small for masters + assert: + that: ansible_memtotal_mb >= minimal_master_memory_mb + when: + - not ignore_assert_errors + - inventory_hostname in groups['kube_control_plane'] + +- name: Stop if memory is too small for nodes + assert: + that: ansible_memtotal_mb >= minimal_node_memory_mb + when: + - not ignore_assert_errors + - inventory_hostname in groups['kube_node'] + +# This assertion will fail on the safe side: One can indeed schedule more pods +# on a node than the CIDR-range has space for when additional pods use the host +# network namespace. It is impossible to ascertain the number of such pods at +# provisioning time, so to establish a guarantee, we factor these out. +# NOTICE: the check blatantly ignores the inet6-case +- name: Guarantee that enough network address space is available for all pods + assert: + that: "{{ (kubelet_max_pods | default(110)) | int <= (2 ** (32 - kube_network_node_prefix | int)) - 2 }}" + msg: "Do not schedule more pods on a node than inet addresses are available." + when: + - not ignore_assert_errors + - inventory_hostname in groups['k8s_cluster'] + - kube_network_node_prefix is defined + - kube_network_plugin != 'calico' + +- name: Stop if ip var does not match local ips + assert: + that: (ip in ansible_all_ipv4_addresses) or (ip in ansible_all_ipv6_addresses) + msg: "IPv4: '{{ ansible_all_ipv4_addresses }}' and IPv6: '{{ ansible_all_ipv6_addresses }}' do not contain '{{ ip }}'" + when: + - not ignore_assert_errors + - ip is defined + +- name: Ensure ping package + package: + # noqa: jinja[spacing] + name: >- + {%- if ansible_os_family == 'Debian' -%} + iputils-ping + {%- else -%} + iputils + {%- endif -%} + state: present + when: + - access_ip is defined + - not ignore_assert_errors + - ping_access_ip + - not is_fedora_coreos + - not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + +- name: Stop if access_ip is not pingable + command: ping -c1 {{ access_ip }} + when: + - access_ip is defined + - not ignore_assert_errors + - ping_access_ip + changed_when: false + +- name: Stop if RBAC is not enabled when dashboard is enabled + assert: + that: rbac_enabled + when: + - dashboard_enabled + - not ignore_assert_errors + +- name: Stop if RBAC is not enabled when OCI cloud controller is enabled + assert: + that: rbac_enabled + when: + - cloud_provider is defined and cloud_provider == "oci" + - not ignore_assert_errors + +- name: Stop if kernel version is too low + assert: + that: ansible_kernel.split('-')[0] is version('4.9.17', '>=') + when: + - kube_network_plugin == 'cilium' or cilium_deploy_additionally | default(false) | bool + - not ignore_assert_errors + +- name: Stop if bad hostname + assert: + that: inventory_hostname is match("[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$") + msg: "Hostname must consist of lower case alphanumeric characters, '.' or '-', and must start and end with an alphanumeric character" + when: not ignore_assert_errors + +- name: Check cloud_provider value + assert: + that: cloud_provider in ['gce', 'aws', 'azure', 'openstack', 'vsphere', 'oci', 'external'] + msg: "If set the 'cloud_provider' var must be set either to 'gce', 'aws', 'azure', 'openstack', 'vsphere', 'oci' or 'external'" + when: + - cloud_provider is defined + - not ignore_assert_errors + tags: + - cloud-provider + - facts + +- name: "Check that kube_service_addresses is a network range" + assert: + that: + - kube_service_addresses | ipaddr('net') + msg: "kube_service_addresses = '{{ kube_service_addresses }}' is not a valid network range" + run_once: yes + +- name: "Check that kube_pods_subnet is a network range" + assert: + that: + - kube_pods_subnet | ipaddr('net') + msg: "kube_pods_subnet = '{{ kube_pods_subnet }}' is not a valid network range" + run_once: yes + +- name: "Check that kube_pods_subnet does not collide with kube_service_addresses" + assert: + that: + - kube_pods_subnet | ipaddr(kube_service_addresses) | string == 'None' + msg: "kube_pods_subnet cannot be the same network segment as kube_service_addresses" + run_once: yes + +- name: "Check that IP range is enough for the nodes" + assert: + that: + - 2 ** (kube_network_node_prefix - kube_pods_subnet | ipaddr('prefix')) >= groups['k8s_cluster'] | length + msg: "Not enough IPs are available for the desired node count." + when: kube_network_plugin != 'calico' + run_once: yes + +- name: Stop if unknown dns mode + assert: + that: dns_mode in ['coredns', 'coredns_dual', 'manual', 'none'] + msg: "dns_mode can only be 'coredns', 'coredns_dual', 'manual' or 'none'" + when: dns_mode is defined + run_once: true + +- name: Stop if unknown kube proxy mode + assert: + that: kube_proxy_mode in ['iptables', 'ipvs'] + msg: "kube_proxy_mode can only be 'iptables' or 'ipvs'" + when: kube_proxy_mode is defined + run_once: true + +- name: Stop if unknown cert_management + assert: + that: cert_management | d('script') in ['script', 'none'] + msg: "cert_management can only be 'script' or 'none'" + run_once: true + +- name: Stop if unknown resolvconf_mode + assert: + that: resolvconf_mode in ['docker_dns', 'host_resolvconf', 'none'] + msg: "resolvconf_mode can only be 'docker_dns', 'host_resolvconf' or 'none'" + when: resolvconf_mode is defined + run_once: true + +- name: Stop if etcd deployment type is not host, docker or kubeadm + assert: + that: etcd_deployment_type in ['host', 'docker', 'kubeadm'] + msg: "The etcd deployment type, 'etcd_deployment_type', must be host, docker or kubeadm" + when: + - inventory_hostname in groups.get('etcd',[]) + +- name: Stop if container manager is not docker, crio or containerd + assert: + that: container_manager in ['docker', 'crio', 'containerd'] + msg: "The container manager, 'container_manager', must be docker, crio or containerd" + run_once: true + +- name: Stop if etcd deployment type is not host or kubeadm when container_manager != docker + assert: + that: etcd_deployment_type in ['host', 'kubeadm'] + msg: "The etcd deployment type, 'etcd_deployment_type', must be host or kubeadm when container_manager is not docker" + when: + - inventory_hostname in groups.get('etcd',[]) + - container_manager != 'docker' + +# TODO: Clean this task up when we drop backward compatibility support for `etcd_kubeadm_enabled` +- name: Stop if etcd deployment type is not host or kubeadm when container_manager != docker and etcd_kubeadm_enabled is not defined + run_once: yes + when: etcd_kubeadm_enabled is defined + block: + - name: Warn the user if they are still using `etcd_kubeadm_enabled` + debug: + msg: > + "WARNING! => `etcd_kubeadm_enabled` is deprecated and will be removed in a future release. + You can set `etcd_deployment_type` to `kubeadm` instead of setting `etcd_kubeadm_enabled` to `true`." + changed_when: true + + - name: Stop if `etcd_kubeadm_enabled` is defined and `etcd_deployment_type` is not `kubeadm` or `host` + assert: + that: etcd_deployment_type == 'kubeadm' + msg: > + It is not possible to use `etcd_kubeadm_enabled` when `etcd_deployment_type` is set to {{ etcd_deployment_type }}. + Unset the `etcd_kubeadm_enabled` variable and set `etcd_deployment_type` to desired deployment type (`host`, `kubeadm`, `docker`) instead." + when: etcd_kubeadm_enabled + +- name: Stop if download_localhost is enabled but download_run_once is not + assert: + that: download_run_once + msg: "download_localhost requires enable download_run_once" + when: download_localhost + +- name: Stop if kata_containers_enabled is enabled when container_manager is docker + assert: + that: container_manager != 'docker' + msg: "kata_containers_enabled support only for containerd and crio-o. See https://github.com/kata-containers/documentation/blob/1.11.4/how-to/run-kata-with-k8s.md#install-a-cri-implementation for details" + when: kata_containers_enabled + +- name: Stop if gvisor_enabled is enabled when container_manager is not containerd + assert: + that: container_manager == 'containerd' + msg: "gvisor_enabled support only compatible with containerd. See https://github.com/kubernetes-sigs/kubespray/issues/7650 for details" + when: gvisor_enabled + +- name: Stop if download_localhost is enabled for Flatcar Container Linux + assert: + that: ansible_os_family not in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + msg: "download_run_once not supported for Flatcar Container Linux" + when: download_run_once or download_force_cache + +- name: Ensure minimum containerd version + assert: + that: containerd_version is version(containerd_min_version_required, '>=') + msg: "containerd_version is too low. Minimum version {{ containerd_min_version_required }}" + run_once: yes + when: + - containerd_version not in ['latest', 'edge', 'stable'] + - container_manager == 'containerd' + +- name: Stop if using deprecated containerd_config variable + assert: + that: containerd_config is not defined + msg: "Variable containerd_config is now deprecated. See https://github.com/kubernetes-sigs/kubespray/blob/master/inventory/sample/group_vars/all/containerd.yml for details." + when: + - containerd_config is defined + - not ignore_assert_errors + +- name: Stop if auto_renew_certificates is enabled when certificates are managed externally (kube_external_ca_mode is true) + assert: + that: not auto_renew_certificates + msg: "Variable auto_renew_certificates must be disabled when CA are managed externally: kube_external_ca_mode = true" + when: + - kube_external_ca_mode + - not ignore_assert_errors + +- name: Stop if using deprecated comma separated list for admission plugins + assert: + that: "',' not in kube_apiserver_enable_admission_plugins[0]" + msg: "Comma-separated list for kube_apiserver_enable_admission_plugins is now deprecated, use separate list items for each plugin." + when: + - kube_apiserver_enable_admission_plugins is defined + - kube_apiserver_enable_admission_plugins | length > 0 diff --git a/kubespray/roles/kubernetes/preinstall/tasks/0050-create_directories.yml b/kubespray/roles/kubernetes/preinstall/tasks/0050-create_directories.yml new file mode 100644 index 0000000..f773989 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/0050-create_directories.yml @@ -0,0 +1,119 @@ +--- +- name: Create kubernetes directories + file: + path: "{{ item }}" + state: directory + owner: "{{ kube_owner }}" + mode: 0755 + when: inventory_hostname in groups['k8s_cluster'] + become: true + tags: + - kubelet + - k8s-secrets + - kube-controller-manager + - kube-apiserver + - bootstrap-os + - apps + - network + - master + - node + with_items: + - "{{ kube_config_dir }}" + - "{{ kube_manifest_dir }}" + - "{{ kube_script_dir }}" + - "{{ kubelet_flexvolumes_plugins_dir }}" + +- name: Create other directories of root owner + file: + path: "{{ item }}" + state: directory + owner: root + mode: 0755 + when: inventory_hostname in groups['k8s_cluster'] + become: true + tags: + - kubelet + - k8s-secrets + - kube-controller-manager + - kube-apiserver + - bootstrap-os + - apps + - network + - master + - node + with_items: + - "{{ kube_cert_dir }}" + - "{{ bin_dir }}" + +- name: Check if kubernetes kubeadm compat cert dir exists + stat: + path: "{{ kube_cert_compat_dir }}" + get_attributes: no + get_checksum: no + get_mime: no + register: kube_cert_compat_dir_check + when: + - inventory_hostname in groups['k8s_cluster'] + - kube_cert_dir != kube_cert_compat_dir + +- name: Create kubernetes kubeadm compat cert dir (kubernetes/kubeadm issue 1498) + file: + src: "{{ kube_cert_dir }}" + dest: "{{ kube_cert_compat_dir }}" + state: link + mode: 0755 + when: + - inventory_hostname in groups['k8s_cluster'] + - kube_cert_dir != kube_cert_compat_dir + - not kube_cert_compat_dir_check.stat.exists + +- name: Create cni directories + file: + path: "{{ item }}" + state: directory + owner: "{{ kube_owner }}" + mode: 0755 + with_items: + - "/etc/cni/net.d" + - "/opt/cni/bin" + when: + - kube_network_plugin in ["calico", "weave", "flannel", "cilium", "kube-ovn", "kube-router", "macvlan"] + - inventory_hostname in groups['k8s_cluster'] + tags: + - network + - cilium + - calico + - weave + - kube-ovn + - kube-router + - bootstrap-os + +- name: Create calico cni directories + file: + path: "{{ item }}" + state: directory + owner: "{{ kube_owner }}" + mode: 0755 + with_items: + - "/var/lib/calico" + when: + - kube_network_plugin == "calico" + - inventory_hostname in groups['k8s_cluster'] + tags: + - network + - calico + - bootstrap-os + +- name: Create local volume provisioner directories + file: + path: "{{ local_volume_provisioner_storage_classes[item].host_dir }}" + state: directory + owner: root + group: root + mode: "{{ local_volume_provisioner_directory_mode }}" + with_items: "{{ local_volume_provisioner_storage_classes.keys() | list }}" + when: + - inventory_hostname in groups['k8s_cluster'] + - local_volume_provisioner_enabled + tags: + - persistent_volumes diff --git a/kubespray/roles/kubernetes/preinstall/tasks/0060-resolvconf.yml b/kubespray/roles/kubernetes/preinstall/tasks/0060-resolvconf.yml new file mode 100644 index 0000000..da5fc85 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/0060-resolvconf.yml @@ -0,0 +1,58 @@ +--- +- name: Create temporary resolveconf cloud init file + command: cp -f /etc/resolv.conf "{{ resolvconffile }}" + when: ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + +- name: Add domain/search/nameservers/options to resolv.conf + blockinfile: + path: "{{ resolvconffile }}" + block: |- + {% for item in [domainentry] + [searchentries] -%} + {{ item }} + {% endfor %} + {% for item in nameserverentries.split(',') %} + nameserver {{ item }} + {% endfor %} + options ndots:{{ ndots }} timeout:{{ dns_timeout | default('2') }} attempts:{{ dns_attempts | default('2') }} + state: present + insertbefore: BOF + create: yes + backup: "{{ not resolvconf_stat.stat.islnk }}" + marker: "# Ansible entries {mark}" + mode: 0644 + notify: Preinstall | propagate resolvconf to k8s components + +- name: Remove search/domain/nameserver options before block + replace: + path: "{{ item[0] }}" + regexp: '^{{ item[1] }}[^#]*(?=# Ansible entries BEGIN)' + backup: "{{ not resolvconf_stat.stat.islnk }}" + with_nested: + - "{{ [resolvconffile, base | default(''), head | default('')] | difference(['']) }}" + - [ 'search\s', 'nameserver\s', 'domain\s', 'options\s' ] + notify: Preinstall | propagate resolvconf to k8s components + +- name: Remove search/domain/nameserver options after block + replace: + path: "{{ item[0] }}" + regexp: '(# Ansible entries END\n(?:(?!^{{ item[1] }}).*\n)*)(?:^{{ item[1] }}.*\n?)+' + replace: '\1' + backup: "{{ not resolvconf_stat.stat.islnk }}" + with_nested: + - "{{ [resolvconffile, base | default(''), head | default('')] | difference(['']) }}" + - [ 'search\s', 'nameserver\s', 'domain\s', 'options\s' ] + notify: Preinstall | propagate resolvconf to k8s components + +- name: Get temporary resolveconf cloud init file content + command: cat {{ resolvconffile }} + register: cloud_config + when: ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + +- name: Persist resolvconf cloud init file + template: + dest: "{{ resolveconf_cloud_init_conf }}" + src: resolvconf.j2 + owner: root + mode: 0644 + notify: Preinstall | update resolvconf for Flatcar Container Linux by Kinvolk + when: ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] diff --git a/kubespray/roles/kubernetes/preinstall/tasks/0061-systemd-resolved.yml b/kubespray/roles/kubernetes/preinstall/tasks/0061-systemd-resolved.yml new file mode 100644 index 0000000..3811358 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/0061-systemd-resolved.yml @@ -0,0 +1,9 @@ +--- +- name: Write resolved.conf + template: + src: resolved.conf.j2 + dest: /etc/systemd/resolved.conf + owner: root + group: root + mode: 0644 + notify: Preinstall | Restart systemd-resolved diff --git a/kubespray/roles/kubernetes/preinstall/tasks/0062-networkmanager-unmanaged-devices.yml b/kubespray/roles/kubernetes/preinstall/tasks/0062-networkmanager-unmanaged-devices.yml new file mode 100644 index 0000000..44d6191 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/0062-networkmanager-unmanaged-devices.yml @@ -0,0 +1,28 @@ +--- +- name: NetworkManager | Ensure NetworkManager conf.d dir + file: + path: "/etc/NetworkManager/conf.d" + state: directory + recurse: yes + +- name: NetworkManager | Prevent NetworkManager from managing Calico interfaces (cali*/tunl*/vxlan.calico) + copy: + content: | + [keyfile] + unmanaged-devices+=interface-name:cali*;interface-name:tunl*;interface-name:vxlan.calico;interface-name:vxlan-v6.calico + dest: /etc/NetworkManager/conf.d/calico.conf + mode: 0644 + when: + - kube_network_plugin == "calico" + notify: Preinstall | reload NetworkManager + +# TODO: add other network_plugin interfaces + +- name: NetworkManager | Prevent NetworkManager from managing K8S interfaces (kube-ipvs0/nodelocaldns) + copy: + content: | + [keyfile] + unmanaged-devices+=interface-name:kube-ipvs0;interface-name:nodelocaldns + dest: /etc/NetworkManager/conf.d/k8s.conf + mode: 0644 + notify: Preinstall | reload NetworkManager diff --git a/kubespray/roles/kubernetes/preinstall/tasks/0063-networkmanager-dns.yml b/kubespray/roles/kubernetes/preinstall/tasks/0063-networkmanager-dns.yml new file mode 100644 index 0000000..e155f0a --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/0063-networkmanager-dns.yml @@ -0,0 +1,37 @@ +--- +- name: NetworkManager | Add nameservers to NM configuration + community.general.ini_file: + path: /etc/NetworkManager/conf.d/dns.conf + section: global-dns-domain-* + option: servers + value: "{{ nameserverentries }}" + mode: '0600' + backup: yes + when: + - nameserverentries != "127.0.0.53" or systemd_resolved_enabled.rc != 0 + notify: Preinstall | update resolvconf for networkmanager + +- name: Set default dns if remove_default_searchdomains is false + set_fact: + default_searchdomains: ["default.svc.{{ dns_domain }}", "svc.{{ dns_domain }}"] + when: not remove_default_searchdomains | default() | bool or (remove_default_searchdomains | default() | bool and searchdomains | default([]) | length==0) + +- name: NetworkManager | Add DNS search to NM configuration + community.general.ini_file: + path: /etc/NetworkManager/conf.d/dns.conf + section: global-dns + option: searches + value: "{{ (default_searchdomains | default([]) + searchdomains | default([])) | join(',') }}" + mode: '0600' + backup: yes + notify: Preinstall | update resolvconf for networkmanager + +- name: NetworkManager | Add DNS options to NM configuration + community.general.ini_file: + path: /etc/NetworkManager/conf.d/dns.conf + section: global-dns + option: options + value: "ndots:{{ ndots }},timeout:{{ dns_timeout | default('2') }},attempts:{{ dns_attempts | default('2') }}" + mode: '0600' + backup: yes + notify: Preinstall | update resolvconf for networkmanager diff --git a/kubespray/roles/kubernetes/preinstall/tasks/0070-system-packages.yml b/kubespray/roles/kubernetes/preinstall/tasks/0070-system-packages.yml new file mode 100644 index 0000000..ccfb490 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/0070-system-packages.yml @@ -0,0 +1,99 @@ +--- +- name: Update package management cache (zypper) - SUSE + command: zypper -n --gpg-auto-import-keys ref + register: make_cache_output + until: make_cache_output is succeeded + retries: 4 + delay: "{{ retry_stagger | random + 3 }}" + when: + - ansible_pkg_mgr == 'zypper' + tags: bootstrap-os + +- name: Add debian 10 required repos + when: + - ansible_distribution == "Debian" + - ansible_distribution_version == "10" + tags: + - bootstrap-os + block: + - name: Add Debian Backports apt repo + apt_repository: + repo: "deb http://deb.debian.org/debian {{ ansible_distribution_release }}-backports main" + state: present + filename: debian-backports + + - name: Set libseccomp2 pin priority to apt_preferences on Debian buster + copy: + content: | + Package: libseccomp2 + Pin: release a={{ ansible_distribution_release }}-backports + Pin-Priority: 1001 + dest: "/etc/apt/preferences.d/libseccomp2" + owner: "root" + mode: 0644 + +- name: Update package management cache (APT) + apt: + update_cache: yes + cache_valid_time: 3600 + when: ansible_os_family == "Debian" + tags: + - bootstrap-os + +- name: Remove legacy docker repo file + file: + path: "{{ yum_repo_dir }}/docker.repo" + state: absent + when: + - ansible_os_family == "RedHat" + - not is_fedora_coreos + +- name: Install python3-dnf for latest RedHat versions + command: dnf install -y python3-dnf + register: dnf_task_result + until: dnf_task_result is succeeded + retries: 4 + delay: "{{ retry_stagger | random + 3 }}" + when: + - ansible_distribution == "Fedora" + - ansible_distribution_major_version | int >= 30 + - not is_fedora_coreos + changed_when: False + tags: + - bootstrap-os + +- name: Install epel-release on RHEL derivatives + package: + name: epel-release + state: present + when: + - ansible_os_family == "RedHat" + - not is_fedora_coreos + - epel_enabled | bool + tags: + - bootstrap-os + +- name: Update common_required_pkgs with ipvsadm when kube_proxy_mode is ipvs + set_fact: + common_required_pkgs: "{{ common_required_pkgs | default([]) + ['ipvsadm', 'ipset'] }}" + when: kube_proxy_mode == 'ipvs' + +- name: Install packages requirements + package: + name: "{{ required_pkgs | default([]) | union(common_required_pkgs | default([])) }}" + state: present + register: pkgs_task_result + until: pkgs_task_result is succeeded + retries: "{{ pkg_install_retries }}" + delay: "{{ retry_stagger | random + 3 }}" + when: not (ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk", "ClearLinux"] or is_fedora_coreos) + tags: + - bootstrap-os + +- name: Install ipvsadm for ClearLinux + package: + name: ipvsadm + state: present + when: + - ansible_os_family in ["ClearLinux"] + - kube_proxy_mode == 'ipvs' diff --git a/kubespray/roles/kubernetes/preinstall/tasks/0080-system-configurations.yml b/kubespray/roles/kubernetes/preinstall/tasks/0080-system-configurations.yml new file mode 100644 index 0000000..87fb176 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/0080-system-configurations.yml @@ -0,0 +1,146 @@ +--- +# Todo : selinux configuration +- name: Confirm selinux deployed + stat: + path: /etc/selinux/config + get_attributes: no + get_checksum: no + get_mime: no + when: + - ansible_os_family == "RedHat" + - "'Amazon' not in ansible_distribution" + register: slc + +- name: Set selinux policy + ansible.posix.selinux: + policy: targeted + state: "{{ preinstall_selinux_state }}" + when: + - ansible_os_family == "RedHat" + - "'Amazon' not in ansible_distribution" + - slc.stat.exists + changed_when: False + tags: + - bootstrap-os + +- name: Disable IPv6 DNS lookup + lineinfile: + dest: /etc/gai.conf + line: "precedence ::ffff:0:0/96 100" + state: present + create: yes + backup: yes + mode: 0644 + when: + - disable_ipv6_dns + - not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + tags: + - bootstrap-os + +- name: Clean previously used sysctl file locations + file: + path: "/etc/sysctl.d/{{ item }}" + state: absent + with_items: + - ipv4-ip_forward.conf + - bridge-nf-call.conf + +- name: Stat sysctl file configuration + stat: + path: "{{ sysctl_file_path }}" + get_attributes: no + get_checksum: no + get_mime: no + register: sysctl_file_stat + tags: + - bootstrap-os + +- name: Change sysctl file path to link source if linked + set_fact: + sysctl_file_path: "{{ sysctl_file_stat.stat.lnk_source }}" + when: + - sysctl_file_stat.stat.islnk is defined + - sysctl_file_stat.stat.islnk + tags: + - bootstrap-os + +- name: Make sure sysctl file path folder exists + file: + name: "{{ sysctl_file_path | dirname }}" + state: directory + mode: 0755 + +- name: Enable ip forwarding + ansible.posix.sysctl: + sysctl_file: "{{ sysctl_file_path }}" + name: net.ipv4.ip_forward + value: "1" + state: present + reload: yes + +- name: Enable ipv6 forwarding + ansible.posix.sysctl: + sysctl_file: "{{ sysctl_file_path }}" + name: net.ipv6.conf.all.forwarding + value: "1" + state: present + reload: yes + when: enable_dual_stack_networks | bool + +- name: Check if we need to set fs.may_detach_mounts + stat: + path: /proc/sys/fs/may_detach_mounts + get_attributes: no + get_checksum: no + get_mime: no + register: fs_may_detach_mounts + ignore_errors: true # noqa ignore-errors + +- name: Set fs.may_detach_mounts if needed + ansible.posix.sysctl: + sysctl_file: "{{ sysctl_file_path }}" + name: fs.may_detach_mounts + value: 1 + state: present + reload: yes + when: fs_may_detach_mounts.stat.exists | d(false) + +- name: Ensure kube-bench parameters are set + ansible.posix.sysctl: + sysctl_file: "{{ sysctl_file_path }}" + name: "{{ item.name }}" + value: "{{ item.value }}" + state: present + reload: yes + with_items: + - { name: kernel.keys.root_maxbytes, value: 25000000 } + - { name: kernel.keys.root_maxkeys, value: 1000000 } + - { name: kernel.panic, value: 10 } + - { name: kernel.panic_on_oops, value: 1 } + - { name: vm.overcommit_memory, value: 1 } + - { name: vm.panic_on_oom, value: 0 } + when: kubelet_protect_kernel_defaults | bool + +- name: Check dummy module + community.general.modprobe: + name: dummy + state: present + params: 'numdummies=0' + when: enable_nodelocaldns + +- name: Set additional sysctl variables + ansible.posix.sysctl: + sysctl_file: "{{ sysctl_file_path }}" + name: "{{ item.name }}" + value: "{{ item.value }}" + state: present + reload: yes + with_items: "{{ additional_sysctl }}" + +- name: Disable fapolicyd service + failed_when: false + systemd: + name: fapolicyd + state: stopped + enabled: false + when: disable_fapolicyd diff --git a/kubespray/roles/kubernetes/preinstall/tasks/0081-ntp-configurations.yml b/kubespray/roles/kubernetes/preinstall/tasks/0081-ntp-configurations.yml new file mode 100644 index 0000000..ad00df3 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/0081-ntp-configurations.yml @@ -0,0 +1,87 @@ +--- +- name: Ensure NTP package + package: + name: + - "{{ ntp_package }}" + state: present + when: + - not is_fedora_coreos + - not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + +- name: Disable systemd-timesyncd + service: + name: systemd-timesyncd.service + enabled: false + state: stopped + failed_when: false + +- name: Set fact NTP settings + set_fact: + # noqa: jinja[spacing] + ntp_config_file: >- + {% if ntp_package == "ntp" -%} + /etc/ntp.conf + {%- elif ansible_os_family in ['RedHat', 'Suse'] -%} + /etc/chrony.conf + {%- else -%} + /etc/chrony/chrony.conf + {%- endif -%} + # noqa: jinja[spacing] + ntp_service_name: >- + {% if ntp_package == "chrony" -%} + chronyd + {%- elif ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk", "RedHat"] -%} + ntpd + {%- else -%} + ntp + {%- endif %} + +- name: Generate NTP configuration file. + template: + src: "{{ ntp_config_file | basename }}.j2" + dest: "{{ ntp_config_file }}" + mode: 0644 + notify: Preinstall | restart ntp + when: + - ntp_manage_config + +- name: Stop the NTP Deamon For Sync Immediately # `ntpd -gq`,`chronyd -q` requires the ntp daemon stop + service: + name: "{{ ntp_service_name }}" + state: stopped + when: + - ntp_force_sync_immediately + +- name: Force Sync NTP Immediately + # noqa: jinja[spacing] + command: >- + timeout -k 60s 60s + {% if ntp_package == "ntp" -%} + ntpd -gq + {%- else -%} + chronyd -q + {%- endif -%} + when: + - ntp_force_sync_immediately + +- name: Ensure NTP service is started and enabled + service: + name: "{{ ntp_service_name }}" + state: started + enabled: true + +- name: Ensure tzdata package + package: + name: + - tzdata + state: present + when: + - ntp_timezone + - not is_fedora_coreos + - not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + +- name: Set timezone + community.general.timezone: + name: "{{ ntp_timezone }}" + when: + - ntp_timezone diff --git a/kubespray/roles/kubernetes/preinstall/tasks/0090-etchosts.yml b/kubespray/roles/kubernetes/preinstall/tasks/0090-etchosts.yml new file mode 100644 index 0000000..6bec169 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/0090-etchosts.yml @@ -0,0 +1,81 @@ +--- +- name: Hosts | create hosts list from inventory + set_fact: + etc_hosts_inventory_block: |- + {% for item in (groups['k8s_cluster'] + groups['etcd'] | default([]) + groups['calico_rr'] | default([])) | unique -%} + {% if 'access_ip' in hostvars[item] or 'ip' in hostvars[item] or 'ansible_default_ipv4' in hostvars[item] -%} + {{ hostvars[item]['access_ip'] | default(hostvars[item]['ip'] | default(hostvars[item]['ansible_default_ipv4']['address'])) }} + {%- if ('ansible_hostname' in hostvars[item] and item != hostvars[item]['ansible_hostname']) %} {{ hostvars[item]['ansible_hostname'] }}.{{ dns_domain }} {{ hostvars[item]['ansible_hostname'] }} {% else %} {{ item }}.{{ dns_domain }} {{ item }} {% endif %} + + {% endif %} + {% endfor %} + delegate_to: localhost + connection: local + delegate_facts: yes + run_once: yes + +- name: Hosts | populate inventory into hosts file + blockinfile: + path: /etc/hosts + block: "{{ hostvars.localhost.etc_hosts_inventory_block }}" + state: "{{ 'present' if populate_inventory_to_hosts_file else 'absent' }}" + create: yes + backup: yes + unsafe_writes: yes + marker: "# Ansible inventory hosts {mark}" + mode: 0644 + +- name: Hosts | populate kubernetes loadbalancer address into hosts file + lineinfile: + dest: /etc/hosts + regexp: ".*{{ apiserver_loadbalancer_domain_name }}$" + line: "{{ loadbalancer_apiserver.address }} {{ apiserver_loadbalancer_domain_name }}" + state: present + backup: yes + unsafe_writes: yes + when: + - populate_loadbalancer_apiserver_to_hosts_file + - loadbalancer_apiserver is defined + - loadbalancer_apiserver.address is defined + +- name: Hosts | Update localhost entries in hosts file + when: populate_localhost_entries_to_hosts_file + block: + - name: Hosts | Retrieve hosts file content + slurp: + src: /etc/hosts + register: etc_hosts_content + + - name: Hosts | Extract existing entries for localhost from hosts file + set_fact: + etc_hosts_localhosts_dict: >- + {%- set splitted = (item | regex_replace('[ \t]+', ' ') | regex_replace('#.*$') | trim).split(' ') -%} + {{ etc_hosts_localhosts_dict | default({}) | combine({splitted[0]: splitted[1::]}) }} + with_items: "{{ (etc_hosts_content['content'] | b64decode).splitlines() }}" + when: + - etc_hosts_content.content is defined + - (item is match('^::1 .*') or item is match('^127.0.0.1 .*')) + + - name: Hosts | Update target hosts file entries dict with required entries + set_fact: + etc_hosts_localhosts_dict_target: >- + {%- set target_entries = (etc_hosts_localhosts_dict | default({})).get(item.key, []) | difference(item.value.get('unexpected', [])) -%} + {{ etc_hosts_localhosts_dict_target | default({}) | combine({item.key: (target_entries + item.value.expected) | unique}) }} + loop: "{{ etc_hosts_localhost_entries | dict2items }}" + + - name: Hosts | Update (if necessary) hosts file + lineinfile: + dest: /etc/hosts + line: "{{ item.key }} {{ item.value | join(' ') }}" + regexp: "^{{ item.key }}.*$" + state: present + backup: yes + unsafe_writes: yes + loop: "{{ etc_hosts_localhosts_dict_target | default({}) | dict2items }}" + +# gather facts to update ansible_fqdn +- name: Update facts + setup: + gather_subset: min + when: + - not dns_late diff --git a/kubespray/roles/kubernetes/preinstall/tasks/0100-dhclient-hooks.yml b/kubespray/roles/kubernetes/preinstall/tasks/0100-dhclient-hooks.yml new file mode 100644 index 0000000..da38147 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/0100-dhclient-hooks.yml @@ -0,0 +1,33 @@ +--- +- name: Configure dhclient to supersede search/domain/nameservers + blockinfile: + block: |- + {% for item in [supersede_domain, supersede_search, supersede_nameserver] -%} + {{ item }} + {% endfor %} + path: "{{ dhclientconffile }}" + create: yes + state: present + insertbefore: BOF + backup: yes + marker: "# Ansible entries {mark}" + mode: 0644 + notify: Preinstall | propagate resolvconf to k8s components + +- name: Configure dhclient hooks for resolv.conf (non-RH) + template: + src: dhclient_dnsupdate.sh.j2 + dest: "{{ dhclienthookfile }}" + owner: root + mode: 0755 + notify: Preinstall | propagate resolvconf to k8s components + when: ansible_os_family not in [ "RedHat", "Suse" ] + +- name: Configure dhclient hooks for resolv.conf (RH-only) + template: + src: dhclient_dnsupdate_rh.sh.j2 + dest: "{{ dhclienthookfile }}" + owner: root + mode: 0755 + notify: Preinstall | propagate resolvconf to k8s components + when: ansible_os_family == "RedHat" diff --git a/kubespray/roles/kubernetes/preinstall/tasks/0110-dhclient-hooks-undo.yml b/kubespray/roles/kubernetes/preinstall/tasks/0110-dhclient-hooks-undo.yml new file mode 100644 index 0000000..024e39f --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/0110-dhclient-hooks-undo.yml @@ -0,0 +1,18 @@ +--- + +# These tasks will undo changes done by kubespray in the past if needed (e.g. when upgrading from kubespray 2.0.x +# or when changing resolvconf_mode) + +- name: Remove kubespray specific config from dhclient config + blockinfile: + path: "{{ dhclientconffile }}" + state: absent + backup: yes + marker: "# Ansible entries {mark}" + notify: Preinstall | propagate resolvconf to k8s components + +- name: Remove kubespray specific dhclient hook + file: + path: "{{ dhclienthookfile }}" + state: absent + notify: Preinstall | propagate resolvconf to k8s components diff --git a/kubespray/roles/kubernetes/preinstall/tasks/0120-growpart-azure-centos-7.yml b/kubespray/roles/kubernetes/preinstall/tasks/0120-growpart-azure-centos-7.yml new file mode 100644 index 0000000..621629f --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/0120-growpart-azure-centos-7.yml @@ -0,0 +1,44 @@ +--- + +# Running growpart seems to be only required on Azure, as other Cloud Providers do this at boot time + +- name: Install growpart + package: + name: cloud-utils-growpart + state: present + +- name: Gather mounts facts + setup: + gather_subset: 'mounts' + +- name: Search root filesystem device + vars: + query: "[?mount=='/'].device" + _root_device: "{{ ansible_mounts | json_query(query) }}" + set_fact: + device: "{{ _root_device | first | regex_replace('([^0-9]+)[0-9]+', '\\1') }}" + partition: "{{ _root_device | first | regex_replace('[^0-9]+([0-9]+)', '\\1') }}" + root_device: "{{ _root_device }}" + +- name: Check if growpart needs to be run + command: growpart -N {{ device }} {{ partition }} + failed_when: False + changed_when: "'NOCHANGE:' not in growpart_needed.stdout" + register: growpart_needed + environment: + LC_ALL: C + +- name: Check fs type + command: file -Ls {{ root_device }} + changed_when: False + register: fs_type + +- name: Run growpart # noqa no-handler + command: growpart {{ device }} {{ partition }} + when: growpart_needed.changed + environment: + LC_ALL: C + +- name: Run xfs_growfs # noqa no-handler + command: xfs_growfs {{ root_device }} + when: growpart_needed.changed and 'XFS' in fs_type.stdout diff --git a/kubespray/roles/kubernetes/preinstall/tasks/main.yml b/kubespray/roles/kubernetes/preinstall/tasks/main.yml new file mode 100644 index 0000000..ee4de5d --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/tasks/main.yml @@ -0,0 +1,148 @@ +--- +# Disable swap +- name: Disable swap + import_tasks: 0010-swapoff.yml + when: + - not dns_late + - kubelet_fail_swap_on + +- name: Set facts + import_tasks: 0020-set_facts.yml + tags: + - resolvconf + - facts + +- name: Check settings + import_tasks: 0040-verify-settings.yml + when: + - not dns_late + tags: + - asserts + +- name: Create directories + import_tasks: 0050-create_directories.yml + when: + - not dns_late + +- name: Apply resolvconf settings + import_tasks: 0060-resolvconf.yml + when: + - dns_mode != 'none' + - resolvconf_mode == 'host_resolvconf' + - systemd_resolved_enabled.rc != 0 + - networkmanager_enabled.rc != 0 + tags: + - bootstrap-os + - resolvconf + +- name: Apply systemd-resolved settings + import_tasks: 0061-systemd-resolved.yml + when: + - dns_mode != 'none' + - resolvconf_mode == 'host_resolvconf' + - systemd_resolved_enabled.rc == 0 + tags: + - bootstrap-os + - resolvconf + +- name: Apply networkmanager unmanaged devices settings + import_tasks: 0062-networkmanager-unmanaged-devices.yml + when: + - networkmanager_enabled.rc == 0 + tags: + - bootstrap-os + +- name: Apply networkmanager DNS settings + import_tasks: 0063-networkmanager-dns.yml + when: + - dns_mode != 'none' + - resolvconf_mode == 'host_resolvconf' + - networkmanager_enabled.rc == 0 + tags: + - bootstrap-os + - resolvconf + +- name: Install required system packages + import_tasks: 0070-system-packages.yml + when: + - not dns_late + tags: + - bootstrap-os + +- name: Apply system configurations + import_tasks: 0080-system-configurations.yml + when: + - not dns_late + tags: + - bootstrap-os + +- name: Configure NTP + import_tasks: 0081-ntp-configurations.yml + when: + - not dns_late + - ntp_enabled + tags: + - bootstrap-os + +- name: Configure /etc/hosts + import_tasks: 0090-etchosts.yml + tags: + - bootstrap-os + - etchosts + +- name: Configure dhclient + import_tasks: 0100-dhclient-hooks.yml + when: + - dns_mode != 'none' + - resolvconf_mode == 'host_resolvconf' + - dhclientconffile is defined + - not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + tags: + - bootstrap-os + - resolvconf + +- name: Configure dhclient dhclient hooks + import_tasks: 0110-dhclient-hooks-undo.yml + when: + - dns_mode != 'none' + - resolvconf_mode != 'host_resolvconf' + - dhclientconffile is defined + - not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + tags: + - bootstrap-os + - resolvconf + +# We need to make sure the network is restarted early enough so that docker can later pick up the correct system +# nameservers and search domains +- name: Flush handlers + meta: flush_handlers + +- name: Check if we are running inside a Azure VM + stat: + path: /var/lib/waagent/ + get_attributes: no + get_checksum: no + get_mime: no + register: azure_check + when: + - not dns_late + tags: + - bootstrap-os + +- name: Grow partition on azure CentOS + import_tasks: 0120-growpart-azure-centos-7.yml + when: + - not dns_late + - azure_check.stat.exists + - ansible_os_family == "RedHat" + - growpart_azure_enabled + tags: + - bootstrap-os + +- name: Run calico checks + include_role: + name: network_plugin/calico + tasks_from: check + when: + - kube_network_plugin == 'calico' + - not ignore_assert_errors diff --git a/kubespray/roles/kubernetes/preinstall/templates/ansible_git.j2 b/kubespray/roles/kubernetes/preinstall/templates/ansible_git.j2 new file mode 100644 index 0000000..abf92a7 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/templates/ansible_git.j2 @@ -0,0 +1,3 @@ +; This file contains the information which identifies the deployment state relative to the git repo +[default] +{{ gitinfo.stdout }} diff --git a/kubespray/roles/kubernetes/preinstall/templates/chrony.conf.j2 b/kubespray/roles/kubernetes/preinstall/templates/chrony.conf.j2 new file mode 100644 index 0000000..7931f43 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/templates/chrony.conf.j2 @@ -0,0 +1,27 @@ +# {{ ansible_managed }} + +# Specify one or more NTP servers. +# Use public servers from the pool.ntp.org project. +# Please consider joining the pool (http://www.pool.ntp.org/join.html). +{% for server in ntp_servers %} +server {{ server }} +{% endfor %} + +# Record the rate at which the system clock gains/losses time. +driftfile /var/lib/chrony/drift + +{% if ntp_tinker_panic is sameas true %} +# Force time sync if the drift exceeds the threshold specified +# Useful for VMs that can be paused and much later resumed. +makestep 1.0 -1 +{% else %} +# Allow the system clock to be stepped in the first three updates +# if its offset is larger than 1 second. +makestep 1.0 3 +{% endif %} + +# Enable kernel synchronization of the real-time clock (RTC). +rtcsync + +# Specify directory for log files. +logdir /var/log/chrony diff --git a/kubespray/roles/kubernetes/preinstall/templates/dhclient_dnsupdate.sh.j2 b/kubespray/roles/kubernetes/preinstall/templates/dhclient_dnsupdate.sh.j2 new file mode 100644 index 0000000..8cf8b81 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/templates/dhclient_dnsupdate.sh.j2 @@ -0,0 +1,13 @@ +#!/bin/sh +# +# Prepend resolver options to /etc/resolv.conf after dhclient` +# regenerates the file. See man (5) resolver for more details. +# +if [ $reason = "BOUND" ]; then + if [ -n "$new_domain_search" -o -n "$new_domain_name_servers" ]; then + RESOLV_CONF=$(cat /etc/resolv.conf | sed -r '/^options (timeout|attempts|ndots).*$/d') + OPTIONS="options timeout:{{ dns_timeout|default('2') }} attempts:{{ dns_attempts|default('2') }} ndots:{{ ndots }}" + + printf "%b\n" "$RESOLV_CONF\n$OPTIONS" > /etc/resolv.conf + fi +fi diff --git a/kubespray/roles/kubernetes/preinstall/templates/dhclient_dnsupdate_rh.sh.j2 b/kubespray/roles/kubernetes/preinstall/templates/dhclient_dnsupdate_rh.sh.j2 new file mode 100644 index 0000000..511839f --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/templates/dhclient_dnsupdate_rh.sh.j2 @@ -0,0 +1,17 @@ +#!/bin/sh +# +# Prepend resolver options to /etc/resolv.conf after dhclient` +# regenerates the file. See man (5) resolver for more details. +# +zdnsupdate_config() { + if [ -n "$new_domain_search" -o -n "$new_domain_name_servers" ]; then + RESOLV_CONF=$(cat /etc/resolv.conf | sed -r '/^options (timeout|attempts|ndots).*$/d') + OPTIONS="options timeout:{{ dns_timeout|default('2') }} attempts:{{ dns_attempts|default('2') }} ndots:{{ ndots }}" + + echo -e "$RESOLV_CONF\n$OPTIONS" > /etc/resolv.conf + fi +} + +zdnsupdate_restore() { + : +} diff --git a/kubespray/roles/kubernetes/preinstall/templates/ntp.conf.j2 b/kubespray/roles/kubernetes/preinstall/templates/ntp.conf.j2 new file mode 100644 index 0000000..abeb899 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/templates/ntp.conf.j2 @@ -0,0 +1,45 @@ +# {{ ansible_managed }} + +# /etc/ntp.conf, configuration for ntpd; see ntp.conf(5) for help + +driftfile {{ ntp_driftfile }} + +{% if ntp_tinker_panic is sameas true %} +# Always reset the clock, even if the new time is more than 1000s away +# from the current system time. Useful for VMs that can be paused +# and much later resumed. +tinker panic 0 +{% endif %} + +# Specify one or more NTP servers. +# Use public servers from the pool.ntp.org project. +# Please consider joining the pool (http://www.pool.ntp.org/join.html). +{% for item in ntp_servers %} +pool {{ item }} +{% endfor %} + +# Access control configuration; see /usr/share/doc/ntp-doc/html/accopt.html for +# details. The web page +# might also be helpful. +# +# Note that "restrict" applies to both servers and clients, so a configuration +# that might be intended to block requests from certain clients could also end +# up blocking replies from your own upstream servers. + +# By default, exchange time with everybody, but don't allow configuration. +restrict -4 default kod notrap nomodify nopeer noquery limited +restrict -6 default kod notrap nomodify nopeer noquery limited + +# Local users may interrogate the ntp server more closely. +{% for item in ntp_restrict %} +restrict {{ item }} +{% endfor %} + +# Needed for adding pool entries +restrict source notrap nomodify noquery + +# Disable the monitoring facility to prevent amplification attacks using ntpdc +# monlist command when default restrict does not include the noquery flag. See +# CVE-2013-5211 for more details. +# Note: Monitoring will not be disabled with the limited restriction flag. +disable monitor diff --git a/kubespray/roles/kubernetes/preinstall/templates/resolvconf.j2 b/kubespray/roles/kubernetes/preinstall/templates/resolvconf.j2 new file mode 100644 index 0000000..807fdd0 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/templates/resolvconf.j2 @@ -0,0 +1,10 @@ +#cloud-config +write_files: + - path: "/etc/resolv.conf" + permissions: "0644" + owner: "root" + content: | + {% for l in cloud_config.stdout_lines %} + {{ l }} + {% endfor %} + # diff --git a/kubespray/roles/kubernetes/preinstall/templates/resolved.conf.j2 b/kubespray/roles/kubernetes/preinstall/templates/resolved.conf.j2 new file mode 100644 index 0000000..0a3b40d --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/templates/resolved.conf.j2 @@ -0,0 +1,21 @@ +[Resolve] +{% if dns_early is sameas true and dns_late is sameas false %} +#DNS= +{% else %} +DNS={{ ([nodelocaldns_ip] if enable_nodelocaldns else coredns_server )| list | join(' ') }} +{% endif %} +FallbackDNS={{ ( upstream_dns_servers|d([]) + nameservers|d([]) + cloud_resolver|d([])) | unique | join(' ') }} +{% if remove_default_searchdomains is sameas false or (remove_default_searchdomains is sameas true and searchdomains|default([])|length==0)%} +Domains={{ ([ 'default.svc.' + dns_domain, 'svc.' + dns_domain ] + searchdomains|default([])) | join(' ') }} +{% else %} +Domains={{ searchdomains|default([]) | join(' ') }} +{% endif %} +#LLMNR=no +#MulticastDNS=no +DNSSEC=no +Cache=no-negative +{% if systemd_resolved_disable_stub_listener | bool %} +DNSStubListener=no +{% else %} +#DNSStubListener=yes +{% endif %} diff --git a/kubespray/roles/kubernetes/preinstall/vars/amazon.yml b/kubespray/roles/kubernetes/preinstall/vars/amazon.yml new file mode 100644 index 0000000..09c645f --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/vars/amazon.yml @@ -0,0 +1,7 @@ +--- +required_pkgs: + - libselinux-python + - device-mapper-libs + - nss + - conntrack-tools + - libseccomp diff --git a/kubespray/roles/kubernetes/preinstall/vars/centos.yml b/kubespray/roles/kubernetes/preinstall/vars/centos.yml new file mode 100644 index 0000000..9b1a874 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/vars/centos.yml @@ -0,0 +1,8 @@ +--- +required_pkgs: + - "{{ ((ansible_distribution_major_version | int) < 8) | ternary('libselinux-python', 'python3-libselinux') }}" + - device-mapper-libs + - nss + - conntrack + - container-selinux + - libseccomp diff --git a/kubespray/roles/kubernetes/preinstall/vars/debian-11.yml b/kubespray/roles/kubernetes/preinstall/vars/debian-11.yml new file mode 100644 index 0000000..59cbc5a --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/vars/debian-11.yml @@ -0,0 +1,10 @@ +--- +required_pkgs: + - python3-apt + - gnupg + - apt-transport-https + - software-properties-common + - conntrack + - iptables + - apparmor + - libseccomp2 diff --git a/kubespray/roles/kubernetes/preinstall/vars/debian-12.yml b/kubespray/roles/kubernetes/preinstall/vars/debian-12.yml new file mode 100644 index 0000000..e0dca4d --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/vars/debian-12.yml @@ -0,0 +1,11 @@ +--- +required_pkgs: + - python3-apt + - gnupg + - apt-transport-https + - software-properties-common + - conntrack + - iptables + - apparmor + - libseccomp2 + - mergerfs diff --git a/kubespray/roles/kubernetes/preinstall/vars/debian.yml b/kubespray/roles/kubernetes/preinstall/vars/debian.yml new file mode 100644 index 0000000..51a2802 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/vars/debian.yml @@ -0,0 +1,9 @@ +--- +required_pkgs: + - python-apt + - aufs-tools + - apt-transport-https + - software-properties-common + - conntrack + - apparmor + - libseccomp2 diff --git a/kubespray/roles/kubernetes/preinstall/vars/fedora.yml b/kubespray/roles/kubernetes/preinstall/vars/fedora.yml new file mode 100644 index 0000000..d69b111 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/vars/fedora.yml @@ -0,0 +1,8 @@ +--- +required_pkgs: + - iptables + - libselinux-python3 + - device-mapper-libs + - conntrack + - container-selinux + - libseccomp diff --git a/kubespray/roles/kubernetes/preinstall/vars/redhat.yml b/kubespray/roles/kubernetes/preinstall/vars/redhat.yml new file mode 100644 index 0000000..9b1a874 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/vars/redhat.yml @@ -0,0 +1,8 @@ +--- +required_pkgs: + - "{{ ((ansible_distribution_major_version | int) < 8) | ternary('libselinux-python', 'python3-libselinux') }}" + - device-mapper-libs + - nss + - conntrack + - container-selinux + - libseccomp diff --git a/kubespray/roles/kubernetes/preinstall/vars/suse.yml b/kubespray/roles/kubernetes/preinstall/vars/suse.yml new file mode 100644 index 0000000..d089ac1 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/vars/suse.yml @@ -0,0 +1,5 @@ +--- +required_pkgs: + - device-mapper + - conntrack-tools + - libseccomp2 diff --git a/kubespray/roles/kubernetes/preinstall/vars/ubuntu.yml b/kubespray/roles/kubernetes/preinstall/vars/ubuntu.yml new file mode 100644 index 0000000..85b3f25 --- /dev/null +++ b/kubespray/roles/kubernetes/preinstall/vars/ubuntu.yml @@ -0,0 +1,8 @@ +--- +required_pkgs: + - python3-apt + - apt-transport-https + - software-properties-common + - conntrack + - apparmor + - libseccomp2 diff --git a/kubespray/roles/kubernetes/tokens/files/kube-gen-token.sh b/kubespray/roles/kubernetes/tokens/files/kube-gen-token.sh new file mode 100644 index 0000000..121b522 --- /dev/null +++ b/kubespray/roles/kubernetes/tokens/files/kube-gen-token.sh @@ -0,0 +1,34 @@ +#!/bin/bash + +# Copyright 2015 The Kubernetes Authors All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +token_dir=${TOKEN_DIR:-/var/srv/kubernetes} +token_file="${token_dir}/known_tokens.csv" + +create_accounts=($@) + +if [ ! -e "${token_file}" ]; then + touch "${token_file}" +fi + +for account in "${create_accounts[@]}"; do + if grep ",${account}," "${token_file}" ; then + continue + fi + token=$(dd if=/dev/urandom bs=128 count=1 2>/dev/null | base64 | tr -d "=+/" | dd bs=32 count=1 2>/dev/null) + echo "${token},${account},${account}" >> "${token_file}" + echo "${token}" > "${token_dir}/${account}.token" + echo "Added ${account}" +done diff --git a/kubespray/roles/kubernetes/tokens/tasks/check-tokens.yml b/kubespray/roles/kubernetes/tokens/tasks/check-tokens.yml new file mode 100644 index 0000000..a157a05 --- /dev/null +++ b/kubespray/roles/kubernetes/tokens/tasks/check-tokens.yml @@ -0,0 +1,41 @@ +--- +- name: "Check_tokens | check if the tokens have already been generated on first master" + stat: + path: "{{ kube_token_dir }}/known_tokens.csv" + get_attributes: no + get_checksum: yes + get_mime: no + delegate_to: "{{ groups['kube_control_plane'][0] }}" + register: known_tokens_master + run_once: true + +- name: "Check_tokens | Set default value for 'sync_tokens' and 'gen_tokens' to false" + set_fact: + sync_tokens: false + gen_tokens: false + +- name: "Check_tokens | Set 'sync_tokens' and 'gen_tokens' to true" + set_fact: + gen_tokens: true + when: not known_tokens_master.stat.exists and kube_token_auth | default(true) + run_once: true + +- name: "Check tokens | check if a cert already exists" + stat: + path: "{{ kube_token_dir }}/known_tokens.csv" + get_attributes: no + get_checksum: yes + get_mime: no + register: known_tokens + +- name: "Check_tokens | Set 'sync_tokens' to true" + set_fact: + sync_tokens: >- + {%- set tokens = {'sync': False} -%} + {%- for server in groups['kube_control_plane'] | intersect(ansible_play_batch) + if (not hostvars[server].known_tokens.stat.exists) or + (hostvars[server].known_tokens.stat.checksum | default('') != known_tokens_master.stat.checksum | default('')) -%} + {%- set _ = tokens.update({'sync': True}) -%} + {%- endfor -%} + {{ tokens.sync }} + run_once: true diff --git a/kubespray/roles/kubernetes/tokens/tasks/gen_tokens.yml b/kubespray/roles/kubernetes/tokens/tasks/gen_tokens.yml new file mode 100644 index 0000000..6ac6b49 --- /dev/null +++ b/kubespray/roles/kubernetes/tokens/tasks/gen_tokens.yml @@ -0,0 +1,63 @@ +--- +- name: Gen_tokens | copy tokens generation script + copy: + src: "kube-gen-token.sh" + dest: "{{ kube_script_dir }}/kube-gen-token.sh" + mode: 0700 + run_once: yes + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: gen_tokens | default(false) + +- name: Gen_tokens | generate tokens for master components + command: "{{ kube_script_dir }}/kube-gen-token.sh {{ item[0] }}-{{ item[1] }}" + environment: + TOKEN_DIR: "{{ kube_token_dir }}" + with_nested: + - [ "system:kubectl" ] + - "{{ groups['kube_control_plane'] }}" + register: gentoken_master + changed_when: "'Added' in gentoken_master.stdout" + run_once: yes + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: gen_tokens | default(false) + +- name: Gen_tokens | generate tokens for node components + command: "{{ kube_script_dir }}/kube-gen-token.sh {{ item[0] }}-{{ item[1] }}" + environment: + TOKEN_DIR: "{{ kube_token_dir }}" + with_nested: + - [ 'system:kubelet' ] + - "{{ groups['kube_node'] }}" + register: gentoken_node + changed_when: "'Added' in gentoken_node.stdout" + run_once: yes + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: gen_tokens | default(false) + +- name: Gen_tokens | Get list of tokens from first master + command: "find {{ kube_token_dir }} -maxdepth 1 -type f" + register: tokens_list + check_mode: no + delegate_to: "{{ groups['kube_control_plane'][0] }}" + run_once: true + when: sync_tokens | default(false) + +- name: Gen_tokens | Gather tokens + shell: "set -o pipefail && tar cfz - {{ tokens_list.stdout_lines | join(' ') }} | base64 --wrap=0" + args: + executable: /bin/bash + register: tokens_data + check_mode: no + delegate_to: "{{ groups['kube_control_plane'][0] }}" + run_once: true + when: sync_tokens | default(false) + +- name: Gen_tokens | Copy tokens on masters + shell: "set -o pipefail && echo '{{ tokens_data.stdout | quote }}' | base64 -d | tar xz -C /" + args: + executable: /bin/bash + when: + - inventory_hostname in groups['kube_control_plane'] + - sync_tokens | default(false) + - inventory_hostname != groups['kube_control_plane'][0] + - tokens_data.stdout diff --git a/kubespray/roles/kubernetes/tokens/tasks/main.yml b/kubespray/roles/kubernetes/tokens/tasks/main.yml new file mode 100644 index 0000000..c9dfd07 --- /dev/null +++ b/kubespray/roles/kubernetes/tokens/tasks/main.yml @@ -0,0 +1,21 @@ +--- + +- name: Check tokens + import_tasks: check-tokens.yml + tags: + - k8s-secrets + - k8s-gen-tokens + - facts + +- name: Make sure the tokens directory exits + file: + path: "{{ kube_token_dir }}" + state: directory + mode: 0644 + group: "{{ kube_cert_group }}" + +- name: Generate tokens + import_tasks: gen_tokens.yml + tags: + - k8s-secrets + - k8s-gen-tokens diff --git a/kubespray/roles/kubespray-defaults/defaults/main.yaml b/kubespray/roles/kubespray-defaults/defaults/main.yaml new file mode 100644 index 0000000..7768ff3 --- /dev/null +++ b/kubespray/roles/kubespray-defaults/defaults/main.yaml @@ -0,0 +1,681 @@ +--- +# Use proxycommand if bastion host is in group all +# This change obseletes editing ansible.cfg file depending on bastion existence +ansible_ssh_common_args: "{% if 'bastion' in groups['all'] %} -o ProxyCommand='ssh -o UserKnownHostsFile=/dev/null -o StrictHostKeyChecking=no -W %h:%p -p {{ hostvars['bastion']['ansible_port'] | default(22) }} {{ hostvars['bastion']['ansible_user'] }}@{{ hostvars['bastion']['ansible_host'] }} {% if ansible_ssh_private_key_file is defined %}-i {{ ansible_ssh_private_key_file }}{% endif %} ' {% endif %}" + +# selinux state +preinstall_selinux_state: permissive + +kube_api_anonymous_auth: true + +# Default value, but will be set to true automatically if detected +is_fedora_coreos: false + +# Swap settings +kubelet_fail_swap_on: true +kubelet_swap_behavior: LimitedSwap + +## Change this to use another Kubernetes version, e.g. a current beta release +kube_version: v1.28.2 + +## The minimum version working +kube_version_min_required: v1.26.0 + +## Kube Proxy mode One of ['iptables', 'ipvs'] +kube_proxy_mode: ipvs + +## The timeout for init first control-plane +kubeadm_init_timeout: 300s + +## List of kubeadm init phases that should be skipped during control plane setup +## By default 'addon/coredns' is skipped +## 'addon/kube-proxy' gets skipped for some network plugins +kubeadm_init_phases_skip_default: [ "addon/coredns" ] +kubeadm_init_phases_skip: >- + {%- if kube_network_plugin == 'kube-router' and (kube_router_run_service_proxy is defined and kube_router_run_service_proxy) -%} + {{ kubeadm_init_phases_skip_default + ["addon/kube-proxy"] }} + {%- elif kube_network_plugin == 'cilium' and (cilium_kube_proxy_replacement is defined and cilium_kube_proxy_replacement == 'strict') -%} + {{ kubeadm_init_phases_skip_default + ["addon/kube-proxy"] }} + {%- elif kube_network_plugin == 'calico' and (calico_bpf_enabled is defined and calico_bpf_enabled) -%} + {{ kubeadm_init_phases_skip_default + ["addon/kube-proxy"] }} + {%- elif kube_proxy_remove is defined and kube_proxy_remove -%} + {{ kubeadm_init_phases_skip_default + ["addon/kube-proxy"] }} + {%- else -%} + {{ kubeadm_init_phases_skip_default }} + {%- endif -%} + +# List of kubeadm phases that should be skipped when joining a new node +# You may need to set this to ['preflight'] for air-gaped deployments to avoid failing connectivity tests. +kubeadm_join_phases_skip_default: [] +kubeadm_join_phases_skip: >- + {{ kubeadm_join_phases_skip_default }} + +# A string slice of values which specify the addresses to use for NodePorts. +# Values may be valid IP blocks (e.g. 1.2.3.0/24, 1.2.3.4/32). +# The default empty string slice ([]) means to use all local addresses. +# kube_proxy_nodeport_addresses_cidr is retained for legacy config +kube_proxy_nodeport_addresses: >- + {%- if kube_proxy_nodeport_addresses_cidr is defined -%} + [{{ kube_proxy_nodeport_addresses_cidr }}] + {%- else -%} + [] + {%- endif -%} + +# Set to true to allow pre-checks to fail and continue deployment +ignore_assert_errors: false + +kube_vip_enabled: false + +# nginx-proxy configure +nginx_config_dir: "/etc/nginx" + +# haproxy configure +haproxy_config_dir: "/etc/haproxy" + +# Directory where the binaries will be installed +bin_dir: /usr/local/bin +docker_bin_dir: /usr/bin +containerd_bin_dir: "{{ bin_dir }}" +etcd_data_dir: /var/lib/etcd +# Where the binaries will be downloaded. +# Note: ensure that you've enough disk space (about 1G) +local_release_dir: "/tmp/releases" +# Random shifts for retrying failed ops like pushing/downloading +retry_stagger: 5 + +# Install epel repo on Centos/RHEL +epel_enabled: false + +# DNS configuration. +# Kubernetes cluster name, also will be used as DNS domain +cluster_name: cluster.local +# Subdomains of DNS domain to be resolved via /etc/resolv.conf for hostnet pods +ndots: 2 +# Default resolv.conf options +docker_dns_options: +- ndots:{{ ndots }} +- timeout:2 +- attempts:2 +# Can be coredns, coredns_dual, manual, or none +dns_mode: coredns + +# Enable nodelocal dns cache +enable_nodelocaldns: true +enable_nodelocaldns_secondary: false +nodelocaldns_ip: 169.254.25.10 +nodelocaldns_health_port: 9254 +nodelocaldns_second_health_port: 9256 +nodelocaldns_bind_metrics_host_ip: false +nodelocaldns_secondary_skew_seconds: 5 + +# Should be set to a cluster IP if using a custom cluster DNS +manual_dns_server: "" + +# Can be host_resolvconf, docker_dns or none +resolvconf_mode: host_resolvconf +# Deploy netchecker app to verify DNS resolve as an HTTP service +deploy_netchecker: false +# Ip address of the kubernetes DNS service (called skydns for historical reasons) +skydns_server: "{{ kube_service_addresses | ipaddr('net') | ipaddr(3) | ipaddr('address') }}" +skydns_server_secondary: "{{ kube_service_addresses | ipaddr('net') | ipaddr(4) | ipaddr('address') }}" +dns_domain: "{{ cluster_name }}" +docker_dns_search_domains: +- 'default.svc.{{ dns_domain }}' +- 'svc.{{ dns_domain }}' + +kube_dns_servers: + coredns: ["{{ skydns_server }}"] + coredns_dual: "{{ [skydns_server] + [skydns_server_secondary] }}" + manual: ["{{ manual_dns_server }}"] + +dns_servers: "{{ kube_dns_servers[dns_mode] }}" + +enable_coredns_k8s_external: false +coredns_k8s_external_zone: k8s_external.local + +enable_coredns_k8s_endpoint_pod_names: false + +# Kubernetes configuration dirs and system namespace. +# Those are where all the additional config stuff goes +# the kubernetes normally puts in /srv/kubernetes. +# This puts them in a sane location and namespace. +# Editing those values will almost surely break something. +kube_config_dir: /etc/kubernetes +kube_script_dir: "{{ bin_dir }}/kubernetes-scripts" +kube_manifest_dir: "{{ kube_config_dir }}/manifests" + +# Kubectl command +# This is for consistency when using kubectl command in roles, and ensure +kubectl: "{{ bin_dir }}/kubectl --kubeconfig {{ kube_config_dir }}/admin.conf" + +# This is where all the cert scripts and certs will be located +kube_cert_dir: "{{ kube_config_dir }}/ssl" + +# compatibility directory for kubeadm +kube_cert_compat_dir: "/etc/kubernetes/pki" + +# This is where all of the bearer tokens will be stored +kube_token_dir: "{{ kube_config_dir }}/tokens" + +# This is the user that owns the cluster installation. +kube_owner: kube + +# This is the group that the cert creation scripts chgrp the +# cert files to. Not really changeable... +kube_cert_group: kube-cert + +# Set to true when the CAs are managed externally. +# When true, disables all tasks manipulating certificates. Ensure before the kubespray run that: +# - Certificates and CAs are present in kube_cert_dir +# - Kubeconfig files are present in kube_config_dir +kube_external_ca_mode: false + +# Cluster Loglevel configuration +kube_log_level: 2 + +# Choose network plugin (cilium, calico, kube-ovn, weave or flannel. Use cni for generic cni plugin) +# Can also be set to 'cloud', which lets the cloud provider setup appropriate routing +kube_network_plugin: calico +kube_network_plugin_multus: false + +# Determines if calico_rr group exists +peer_with_calico_rr: "{{ 'calico_rr' in groups and groups['calico_rr'] | length > 0 }}" + +# Choose data store type for calico: "etcd" or "kdd" (kubernetes datastore) +calico_datastore: "kdd" + +# Kubernetes internal network for services, unused block of space. +kube_service_addresses: 10.233.0.0/18 + +# internal network. When used, it will assign IP +# addresses from this range to individual pods. +# This network must be unused in your network infrastructure! +kube_pods_subnet: 10.233.64.0/18 + +# internal network node size allocation (optional). This is the size allocated +# to each node for pod IP address allocation. Note that the number of pods per node is +# also limited by the kubelet_max_pods variable which defaults to 110. +# +# Example: +# Up to 64 nodes and up to 254 or kubelet_max_pods (the lowest of the two) pods per node: +# - kube_pods_subnet: 10.233.64.0/18 +# - kube_network_node_prefix: 24 +# - kubelet_max_pods: 110 +# +# Example: +# Up to 128 nodes and up to 126 or kubelet_max_pods (the lowest of the two) pods per node: +# - kube_pods_subnet: 10.233.64.0/18 +# - kube_network_node_prefix: 25 +# - kubelet_max_pods: 110 +kube_network_node_prefix: 24 + +# Configure Dual Stack networking (i.e. both IPv4 and IPv6) +enable_dual_stack_networks: false + +# Kubernetes internal network for IPv6 services, unused block of space. +# This is only used if enable_dual_stack_networks is set to true +# This provides 4096 IPv6 IPs +kube_service_addresses_ipv6: fd85:ee78:d8a6:8607::1000/116 + +# Internal network. When used, it will assign IPv6 addresses from this range to individual pods. +# This network must not already be in your network infrastructure! +# This is only used if enable_dual_stack_networks is set to true. +# This provides room for 256 nodes with 254 pods per node. +kube_pods_subnet_ipv6: fd85:ee78:d8a6:8607::1:0000/112 + +# IPv6 subnet size allocated to each for pods. +# This is only used if enable_dual_stack_networks is set to true +# This provides room for 254 pods per node. +kube_network_node_prefix_ipv6: 120 + +# The virtual cluster IP, real host IPs and ports the API Server will be +# listening on. +# NOTE: loadbalancer_apiserver_localhost somewhat alters the final API enpdoint +# access IP value (automatically evaluated below) +kube_apiserver_ip: "{{ kube_service_addresses | ipaddr('net') | ipaddr(1) | ipaddr('address') }}" + +# NOTE: If you specific address/interface and use loadbalancer_apiserver_localhost +# loadbalancer_apiserver_localhost (nginx/haproxy) will deploy on masters on 127.0.0.1:{{ loadbalancer_apiserver_port | default(kube_apiserver_port) }} too. +kube_apiserver_bind_address: 0.0.0.0 + +# https +kube_apiserver_port: 6443 + +# If non-empty, will use this string as identification instead of the actual hostname +kube_override_hostname: >- + {%- if cloud_provider is defined and cloud_provider in ['aws'] -%} + {%- else -%} + {{ inventory_hostname }} + {%- endif -%} + +# define kubelet config dir for dynamic kubelet +# kubelet_config_dir: +default_kubelet_config_dir: "{{ kube_config_dir }}/dynamic_kubelet_dir" + +# Aggregator +kube_api_aggregator_routing: false + +# Profiling +kube_profiling: false + +# Graceful Node Shutdown +kubelet_shutdown_grace_period: 60s +# kubelet_shutdown_grace_period_critical_pods should be less than kubelet_shutdown_grace_period +# to give normal pods time to be gracefully evacuated +kubelet_shutdown_grace_period_critical_pods: 20s + +# Whether to deploy the container engine +deploy_container_engine: "{{ inventory_hostname in groups['k8s_cluster'] or etcd_deployment_type == 'docker' }}" + +# Container for runtime +container_manager: containerd + +# Enable Node Resource Interface in containerd or CRI-O. Requires crio_version >= v1.26.0 +# or containerd_version >= 1.7.0. +nri_enabled: false + +# Enable Kata Containers as additional container runtime +# When enabled, it requires `container_manager` different than Docker +kata_containers_enabled: false + +# Enable gVisor as an additional container runtime +# gVisor is only supported with container_manager Docker or containerd +gvisor_enabled: false + +# Enable crun as additional container runtime +# When enabled, it requires container_manager=crio +crun_enabled: false + +# Enable youki as additional container runtime +# When enabled, it requires container_manager=crio +youki_enabled: false + +# Container on localhost (download images when download_localhost is true) +container_manager_on_localhost: "{{ container_manager }}" + +# CRI socket path +cri_socket: >- + {%- if container_manager == 'crio' -%} + unix:///var/run/crio/crio.sock + {%- elif container_manager == 'containerd' -%} + unix:///var/run/containerd/containerd.sock + {%- elif container_manager == 'docker' -%} + unix:///var/run/cri-dockerd.sock + {%- endif -%} + +crio_insecure_registries: [] + +## Uncomment this if you want to force overlay/overlay2 as docker storage driver +## Please note that overlay2 is only supported on newer kernels +# docker_storage_options: -s overlay2 + +## Enable docker_container_storage_setup, it will configure devicemapper driver on Centos7 or RedHat7. +docker_container_storage_setup: false + +## It must be define a disk path for docker_container_storage_setup_devs. +## Otherwise docker-storage-setup will be executed incorrectly. +# docker_container_storage_setup_devs: /dev/vdb + +## Only set this if you have more than 3 nameservers: +## If true Kubespray will only use the first 3, otherwise it will fail +docker_dns_servers_strict: false + +# Path used to store Docker data +docker_daemon_graph: "/var/lib/docker" + +## Used to set docker daemon iptables options to true +docker_iptables_enabled: "false" + +# Docker log options +# Rotate container stderr/stdout logs at 50m and keep last 5 +docker_log_opts: "--log-opt max-size=50m --log-opt max-file=5" + +## A list of insecure docker registries (IP address or domain name), for example +## to allow insecure-registry access to self-hosted registries. Empty by default. +# docker_insecure_registries: +# - mirror.registry.io +# - 172.19.16.11 +docker_insecure_registries: [] + +## A list of additional registry mirrors, for example China registry mirror. Empty by default. +# docker_registry_mirrors: +# - https://registry.docker-cn.com +# - https://mirror.aliyuncs.com +docker_registry_mirrors: [] + +## If non-empty will override default system MounFlags value. +## This option takes a mount propagation flag: shared, slave +## or private, which control whether mounts in the file system +## namespace set up for docker will receive or propagate mounts +## and unmounts. Leave empty for system default +# docker_mount_flags: + +## A string of extra options to pass to the docker daemon. +# docker_options: "" + +## A list of plugins to install using 'docker plugin install --grant-all-permissions' +## Empty by default so no plugins will be installed. +docker_plugins: [] + +# Containerd options - thse are relevant when container_manager == 'containerd' +containerd_use_systemd_cgroup: true + +# Containerd conf default dir +containerd_storage_dir: "/var/lib/containerd" +containerd_state_dir: "/run/containerd" +containerd_systemd_dir: "/etc/systemd/system/containerd.service.d" +containerd_cfg_dir: "/etc/containerd" + +# Settings for containerized control plane (etcd/kubelet/secrets) +# deployment type for legacy etcd mode +etcd_deployment_type: host +cert_management: script + +# Make a copy of kubeconfig on the host that runs Ansible in {{ inventory_dir }}/artifacts +kubeconfig_localhost: false +# Download kubectl onto the host that runs Ansible in {{ bin_dir }} +kubectl_localhost: false + +# Define credentials_dir here so it can be overridden +credentials_dir: "{{ inventory_dir }}/credentials" + +# K8s image pull policy (imagePullPolicy) +k8s_image_pull_policy: IfNotPresent + +# Kubernetes dashboard +# RBAC required. see docs/getting-started.md for access details. +dashboard_enabled: false + +# Addons which can be enabled +helm_enabled: false +krew_enabled: false +registry_enabled: false +metrics_server_enabled: false +enable_network_policy: true +local_path_provisioner_enabled: false +local_volume_provisioner_enabled: false +local_volume_provisioner_directory_mode: 0700 +cinder_csi_enabled: false +aws_ebs_csi_enabled: false +azure_csi_enabled: false +gcp_pd_csi_enabled: false +vsphere_csi_enabled: false +upcloud_csi_enabled: false +csi_snapshot_controller_enabled: false +persistent_volumes_enabled: false +cephfs_provisioner_enabled: false +rbd_provisioner_enabled: false +ingress_nginx_enabled: false +ingress_alb_enabled: false +cert_manager_enabled: false +expand_persistent_volumes: false +metallb_enabled: false +metallb_speaker_enabled: "{{ metallb_enabled }}" +argocd_enabled: false + +## When OpenStack is used, Cinder version can be explicitly specified if autodetection fails (Fixed in 1.9: https://github.com/kubernetes/kubernetes/issues/50461) +# openstack_blockstorage_version: "v1/v2/auto (default)" +openstack_blockstorage_ignore_volume_az: "{{ volume_cross_zone_attachment | default('false') }}" +# set max volumes per node (cinder-csi), default not set +# node_volume_attach_limit: 25 +# Cinder CSI topology, when false volumes can be cross-mounted between availability zones +# cinder_topology: false +# Set Cinder topology zones (can be multiple zones, default not set) +# cinder_topology_zones: +# - nova +cinder_csi_ignore_volume_az: "{{ volume_cross_zone_attachment | default('false') }}" + +## When OpenStack is used, if LBaaSv2 is available you can enable it with the following 2 variables. +openstack_lbaas_enabled: false +# openstack_lbaas_subnet_id: "Neutron subnet ID (not network ID) to create LBaaS VIP" +## To enable automatic floating ip provisioning, specify a subnet. +# openstack_lbaas_floating_network_id: "Neutron network ID (not subnet ID) to get floating IP from, disabled by default" +## Override default LBaaS behavior +# openstack_lbaas_use_octavia: False +# openstack_lbaas_method: "ROUND_ROBIN" +# openstack_lbaas_provider: "haproxy" +openstack_lbaas_create_monitor: "yes" +openstack_lbaas_monitor_delay: "1m" +openstack_lbaas_monitor_timeout: "30s" +openstack_lbaas_monitor_max_retries: "3" +openstack_cacert: "{{ lookup('env', 'OS_CACERT') }}" + +# Default values for the external OpenStack Cloud Controller +external_openstack_lbaas_enabled: true +external_openstack_network_ipv6_disabled: false +external_openstack_network_internal_networks: [] +external_openstack_network_public_networks: [] + +# Default values for the external Hcloud Cloud Controller +external_hcloud_cloud: + hcloud_api_token: "" + token_secret_name: hcloud + + service_account_name: cloud-controller-manager + + controller_image_tag: "latest" + ## A dictionary of extra arguments to add to the openstack cloud controller manager daemonset + ## Format: + ## external_hcloud_cloud.controller_extra_args: + ## arg1: "value1" + ## arg2: "value2" + controller_extra_args: {} + +## List of authorization modes that must be configured for +## the k8s cluster. Only 'AlwaysAllow', 'AlwaysDeny', 'Node' and +## 'RBAC' modes are tested. Order is important. +authorization_modes: ['Node', 'RBAC'] +rbac_enabled: "{{ 'RBAC' in authorization_modes }}" + +# When enabled, API bearer tokens (including service account tokens) can be used to authenticate to the kubelet's HTTPS endpoint +kubelet_authentication_token_webhook: true + +# When enabled, access to the kubelet API requires authorization by delegation to the API server +kubelet_authorization_mode_webhook: true + +# kubelet uses certificates for authenticating to the Kubernetes API +# Automatically generate a new key and request a new certificate from the Kubernetes API as the current certificate approaches expiration +kubelet_rotate_certificates: true +# kubelet can also request a new server certificate from the Kubernetes API +kubelet_rotate_server_certificates: false + +# If set to true, kubelet errors if any of kernel tunables is different than kubelet defaults +kubelet_protect_kernel_defaults: true + +# Set additional sysctl variables to modify Linux kernel variables, for example: +# additional_sysctl: +# - { name: kernel.pid_max, value: 131072 } +# +additional_sysctl: [] + +## List of key=value pairs that describe feature gates for +## the k8s cluster. +kube_feature_gates: [] +kube_apiserver_feature_gates: [] +kube_controller_feature_gates: [] +kube_scheduler_feature_gates: [] +kube_proxy_feature_gates: [] +kubelet_feature_gates: [] +kubeadm_feature_gates: [] + +# Local volume provisioner storage classes +# Levarages Ansibles string to Python datatype casting. Otherwise the dict_key isn't substituted +# see https://github.com/ansible/ansible/issues/17324 +local_volume_provisioner_storage_classes: | + { + "{{ local_volume_provisioner_storage_class | default('local-storage') }}": { + "host_dir": "{{ local_volume_provisioner_base_dir | default('/mnt/disks') }}", + "mount_dir": "{{ local_volume_provisioner_mount_dir | default('/mnt/disks') }}", + "volume_mode": "Filesystem", + "fs_type": "ext4" + + } + } + +# weave's network password for encryption +# if null then no network encryption +# you can use --extra-vars to pass the password in command line +weave_password: EnterPasswordHere + +ssl_ca_dirs: |- + [ + {% if ansible_os_family in ['Flatcar', 'Flatcar Container Linux by Kinvolk'] -%} + '/usr/share/ca-certificates', + {% elif ansible_os_family == 'RedHat' -%} + '/etc/pki/tls', + '/etc/pki/ca-trust', + {% elif ansible_os_family == 'Debian' -%} + '/usr/share/ca-certificates', + {% endif -%} + ] + +# Vars for pointing to kubernetes api endpoints +is_kube_master: "{{ inventory_hostname in groups['kube_control_plane'] }}" +kube_apiserver_count: "{{ groups['kube_control_plane'] | length }}" +kube_apiserver_address: "{{ ip | default(fallback_ips[inventory_hostname]) }}" +kube_apiserver_access_address: "{{ access_ip | default(kube_apiserver_address) }}" +first_kube_control_plane_address: "{{ hostvars[groups['kube_control_plane'][0]]['access_ip'] | default(hostvars[groups['kube_control_plane'][0]]['ip'] | default(fallback_ips[groups['kube_control_plane'][0]])) }}" +loadbalancer_apiserver_localhost: "{{ loadbalancer_apiserver is not defined }}" +loadbalancer_apiserver_type: "nginx" +# applied if only external loadbalancer_apiserver is defined, otherwise ignored +apiserver_loadbalancer_domain_name: "lb-apiserver.kubernetes.local" +kube_apiserver_global_endpoint: |- + {% if loadbalancer_apiserver is defined -%} + https://{{ apiserver_loadbalancer_domain_name }}:{{ loadbalancer_apiserver.port | default(kube_apiserver_port) }} + {%- elif loadbalancer_apiserver_localhost and (loadbalancer_apiserver_port is not defined or loadbalancer_apiserver_port == kube_apiserver_port) -%} + https://localhost:{{ kube_apiserver_port }} + {%- else -%} + https://{{ first_kube_control_plane_address }}:{{ kube_apiserver_port }} + {%- endif %} +kube_apiserver_endpoint: |- + {% if loadbalancer_apiserver is defined -%} + https://{{ apiserver_loadbalancer_domain_name }}:{{ loadbalancer_apiserver.port | default(kube_apiserver_port) }} + {%- elif not is_kube_master and loadbalancer_apiserver_localhost -%} + https://localhost:{{ loadbalancer_apiserver_port | default(kube_apiserver_port) }} + {%- elif is_kube_master -%} + https://{{ kube_apiserver_bind_address | regex_replace('0\.0\.0\.0', '127.0.0.1') }}:{{ kube_apiserver_port }} + {%- else -%} + https://{{ first_kube_control_plane_address }}:{{ kube_apiserver_port }} + {%- endif %} +kube_apiserver_client_cert: "{{ kube_cert_dir }}/ca.crt" +kube_apiserver_client_key: "{{ kube_cert_dir }}/ca.key" + +# Set to true to deploy etcd-events cluster +etcd_events_cluster_enabled: false + +# etcd group can be empty when kubeadm manages etcd +etcd_hosts: "{{ groups['etcd'] | default(groups['kube_control_plane']) }}" + +# Vars for pointing to etcd endpoints +is_etcd_master: "{{ inventory_hostname in groups['etcd'] }}" +etcd_address: "{{ ip | default(fallback_ips[inventory_hostname]) }}" +etcd_access_address: "{{ access_ip | default(etcd_address) }}" +etcd_events_access_address: "{{ access_ip | default(etcd_address) }}" +etcd_peer_url: "https://{{ etcd_access_address }}:2380" +etcd_client_url: "https://{{ etcd_access_address }}:2379" +etcd_events_peer_url: "https://{{ etcd_events_access_address }}:2382" +etcd_events_client_url: "https://{{ etcd_events_access_address }}:2383" +etcd_access_addresses: |- + {% for item in etcd_hosts -%} + https://{{ hostvars[item]['etcd_access_address'] | default(hostvars[item]['ip'] | default(fallback_ips[item])) }}:2379{% if not loop.last %},{% endif %} + {%- endfor %} +etcd_events_access_addresses_list: |- + [ + {% for item in etcd_hosts -%} + 'https://{{ hostvars[item]['etcd_events_access_address'] | default(hostvars[item]['ip'] | default(fallback_ips[item])) }}:2383'{% if not loop.last %},{% endif %} + {%- endfor %} + ] +etcd_metrics_addresses: |- + {% for item in etcd_hosts -%} + https://{{ hostvars[item]['etcd_access_address'] | default(hostvars[item]['ip'] | default(fallback_ips[item])) }}:{{ etcd_metrics_port | default(2381) }}{% if not loop.last %},{% endif %} + {%- endfor %} +etcd_events_access_addresses: "{{ etcd_events_access_addresses_list | join(',') }}" +etcd_events_access_addresses_semicolon: "{{ etcd_events_access_addresses_list | join(';') }}" +# user should set etcd_member_name in inventory/mycluster/hosts.ini +etcd_member_name: |- + {% for host in groups['etcd'] %} + {% if inventory_hostname == host %}{{ hostvars[host].etcd_member_name | default("etcd" + loop.index | string) }}{% endif %} + {% endfor %} +etcd_peer_addresses: |- + {% for item in groups['etcd'] -%} + {{ hostvars[item].etcd_member_name | default("etcd" + loop.index | string) }}=https://{{ hostvars[item].etcd_access_address | default(hostvars[item].ip | default(fallback_ips[item])) }}:2380{% if not loop.last %},{% endif %} + {%- endfor %} +etcd_events_peer_addresses: |- + {% for item in groups['etcd'] -%} + {{ hostvars[item].etcd_member_name | default("etcd" + loop.index | string) }}-events=https://{{ hostvars[item].etcd_events_access_address | default(hostvars[item].ip | default(fallback_ips[item])) }}:2382{% if not loop.last %},{% endif %} + {%- endfor %} + +podsecuritypolicy_enabled: false +etcd_heartbeat_interval: "250" +etcd_election_timeout: "5000" +etcd_snapshot_count: "10000" + +certificates_key_size: 2048 +certificates_duration: 36500 + +etcd_config_dir: /etc/ssl/etcd +etcd_events_data_dir: "/var/lib/etcd-events" +etcd_cert_dir: "{{ etcd_config_dir }}/ssl" + +typha_enabled: false + +calico_apiserver_enabled: false + +_host_architecture_groups: + x86_64: amd64 + aarch64: arm64 + armv7l: arm +host_architecture: >- + {%- if ansible_architecture in _host_architecture_groups -%} + {{ _host_architecture_groups[ansible_architecture] }} + {%- else -%} + {{ ansible_architecture }} + {%- endif -%} + +_host_os_groups: + Linux: linux + Darwin: darwin + Win32NT: windows +host_os: >- + {%- if ansible_system in _host_os_groups -%} + {{ _host_os_groups[ansible_system] }} + {%- else -%} + {{ ansible_system }} + {%- endif -%} + +# Sets the eventRecordQPS parameter in kubelet-config.yaml. The default value is 5 (see types.go) +# Setting it to 0 allows unlimited requests per second. +kubelet_event_record_qps: 5 + +proxy_env_defaults: + http_proxy: "{{ http_proxy | default('') }}" + HTTP_PROXY: "{{ http_proxy | default('') }}" + https_proxy: "{{ https_proxy | default('') }}" + HTTPS_PROXY: "{{ https_proxy | default('') }}" + no_proxy: "{{ no_proxy | default('') }}" + NO_PROXY: "{{ no_proxy | default('') }}" + +# If we use SSL_CERT_FILE: {{ omit }} it cause in value __omit_place_holder__ and break environments +# Combine dict is avoiding the problem with omit placeholder. Maybe it can be better solution? +proxy_env: "{{ proxy_env_defaults | combine({'SSL_CERT_FILE': https_proxy_cert_file}) if https_proxy_cert_file is defined else proxy_env_defaults }}" + +proxy_disable_env: + ALL_PROXY: '' + FTP_PROXY: '' + HTTPS_PROXY: '' + HTTP_PROXY: '' + NO_PROXY: '' + all_proxy: '' + ftp_proxy: '' + http_proxy: '' + https_proxy: '' + no_proxy: '' + +# krew root dir +krew_root_dir: "/usr/local/krew" + +# sysctl_file_path to add sysctl conf to +sysctl_file_path: "/etc/sysctl.d/99-sysctl.conf" + +system_upgrade: false +system_upgrade_reboot: on-upgrade # never, always diff --git a/kubespray/roles/kubespray-defaults/meta/main.yml b/kubespray/roles/kubespray-defaults/meta/main.yml new file mode 100644 index 0000000..88d7024 --- /dev/null +++ b/kubespray/roles/kubespray-defaults/meta/main.yml @@ -0,0 +1,6 @@ +--- +dependencies: + - role: download + skip_downloads: true + tags: + - facts diff --git a/kubespray/roles/kubespray-defaults/tasks/fallback_ips.yml b/kubespray/roles/kubespray-defaults/tasks/fallback_ips.yml new file mode 100644 index 0000000..9aa0ea2 --- /dev/null +++ b/kubespray/roles/kubespray-defaults/tasks/fallback_ips.yml @@ -0,0 +1,33 @@ +--- +# Set 127.0.0.1 as fallback IP if we do not have host facts for host +# ansible_default_ipv4 isn't what you think. +# Thanks https://medium.com/opsops/ansible-default-ipv4-is-not-what-you-think-edb8ab154b10 + +- name: Gather ansible_default_ipv4 from all hosts + setup: + gather_subset: '!all,network' + filter: "ansible_default_ipv4" + delegate_to: "{{ item }}" + delegate_facts: yes + when: hostvars[item].ansible_default_ipv4 is not defined + loop: "{{ (groups['k8s_cluster'] | default([]) + groups['etcd'] | default([]) + groups['calico_rr'] | default([])) | unique }}" + run_once: yes + tags: always + +- name: Create fallback_ips_base + set_fact: + fallback_ips_base: | + --- + {% for item in (groups['k8s_cluster'] | default([]) + groups['etcd'] | default([]) + groups['calico_rr'] | default([])) | unique %} + {% set found = hostvars[item].get('ansible_default_ipv4') %} + {{ item }}: "{{ found.get('address', '127.0.0.1') }}" + {% endfor %} + delegate_to: localhost + connection: local + delegate_facts: yes + become: no + run_once: yes + +- name: Set fallback_ips + set_fact: + fallback_ips: "{{ hostvars.localhost.fallback_ips_base | from_yaml }}" diff --git a/kubespray/roles/kubespray-defaults/tasks/main.yaml b/kubespray/roles/kubespray-defaults/tasks/main.yaml new file mode 100644 index 0000000..ebd9b89 --- /dev/null +++ b/kubespray/roles/kubespray-defaults/tasks/main.yaml @@ -0,0 +1,33 @@ +--- +- name: Configure defaults + debug: + msg: "Check roles/kubespray-defaults/defaults/main.yml" + tags: + - always + +# do not run gather facts when bootstrap-os in roles +- name: Set fallback_ips + import_tasks: fallback_ips.yml + when: + - "'bootstrap-os' not in ansible_play_role_names" + - fallback_ips is not defined + tags: + - always + +- name: Set no_proxy + import_tasks: no_proxy.yml + when: + - "'bootstrap-os' not in ansible_play_role_names" + - http_proxy is defined or https_proxy is defined + - no_proxy is not defined + tags: + - always + +# TODO: Clean this task up when we drop backward compatibility support for `etcd_kubeadm_enabled` +- name: Set `etcd_deployment_type` to "kubeadm" if `etcd_kubeadm_enabled` is true + set_fact: + etcd_deployment_type: kubeadm + when: + - etcd_kubeadm_enabled is defined and etcd_kubeadm_enabled + tags: + - always diff --git a/kubespray/roles/kubespray-defaults/tasks/no_proxy.yml b/kubespray/roles/kubespray-defaults/tasks/no_proxy.yml new file mode 100644 index 0000000..d2d5cc6 --- /dev/null +++ b/kubespray/roles/kubespray-defaults/tasks/no_proxy.yml @@ -0,0 +1,40 @@ +--- +- name: Set no_proxy to all assigned cluster IPs and hostnames + set_fact: + # noqa: jinja[spacing] + no_proxy_prepare: >- + {%- if loadbalancer_apiserver is defined -%} + {{ apiserver_loadbalancer_domain_name | default('') }}, + {{ loadbalancer_apiserver.address | default('') }}, + {%- endif -%} + {%- if no_proxy_exclude_workers | default(false) -%} + {% set cluster_or_master = 'kube_control_plane' %} + {%- else -%} + {% set cluster_or_master = 'k8s_cluster' %} + {%- endif -%} + {%- for item in (groups[cluster_or_master] + groups['etcd'] | default([]) + groups['calico_rr'] | default([])) | unique -%} + {{ hostvars[item]['access_ip'] | default(hostvars[item]['ip'] | default(fallback_ips[item])) }}, + {%- if item != hostvars[item].get('ansible_hostname', '') -%} + {{ hostvars[item]['ansible_hostname'] }}, + {{ hostvars[item]['ansible_hostname'] }}.{{ dns_domain }}, + {%- endif -%} + {{ item }},{{ item }}.{{ dns_domain }}, + {%- endfor -%} + {%- if additional_no_proxy is defined -%} + {{ additional_no_proxy }}, + {%- endif -%} + 127.0.0.1,localhost,{{ kube_service_addresses }},{{ kube_pods_subnet }},svc,svc.{{ dns_domain }} + delegate_to: localhost + connection: local + delegate_facts: yes + become: no + run_once: yes + +- name: Populates no_proxy to all hosts + set_fact: + no_proxy: "{{ hostvars.localhost.no_proxy_prepare }}" + # noqa: jinja[spacing] + proxy_env: "{{ proxy_env | combine({ + 'no_proxy': hostvars.localhost.no_proxy_prepare, + 'NO_PROXY': hostvars.localhost.no_proxy_prepare + }) }}" diff --git a/kubespray/roles/kubespray-defaults/vars/main.yml b/kubespray/roles/kubespray-defaults/vars/main.yml new file mode 100644 index 0000000..c79edf5 --- /dev/null +++ b/kubespray/roles/kubespray-defaults/vars/main.yml @@ -0,0 +1,9 @@ +--- +# Kubespray constants + +kube_proxy_deployed: "{{ 'addon/kube-proxy' not in kubeadm_init_phases_skip }}" + +# The lowest version allowed to upgrade from (same as calico_version in the previous branch) +calico_min_version_required: "v3.19.4" + +containerd_min_version_required: "1.3.7" diff --git a/kubespray/roles/moaroom/defaults/main.yml b/kubespray/roles/moaroom/defaults/main.yml new file mode 100755 index 0000000..a4f390d --- /dev/null +++ b/kubespray/roles/moaroom/defaults/main.yml @@ -0,0 +1,24 @@ +--- +# defaults file for artiference +artiference_namespace: artiference +istio_namespace: artiference-istio-system + +# gateway setting +node_port: 30090 +artiference_host: "artiference.duckdns.org" + +# hosts +grafana_host: "grafana-artiference.duckdns.org" +kibana_host: "kibana.artiference.dudaji.com" +harbor_host: "harbor.artiference.dudaji.com" +minio_host: "minio-artiference.duckdns.org" + +# storage setting +storage_node_name: master # storage node should have large disk space +nfs_path: "/home/artiference_db/{{ artiference_namespace }}" # this is for nfs server +artiference_storageclass: "{{ artiference_namespace }}-nfs" # this is nfs server + +artiference_tag: 0.4.0 +inference_manager_tag: "nhn-apply-sso" +web_tag: "nhn-apply-sso" +docker_origin: dudaji diff --git a/kubespray/roles/moaroom/templates/deploy-control.yml.j2 b/kubespray/roles/moaroom/templates/deploy-control.yml.j2 new file mode 100644 index 0000000..e97ca27 --- /dev/null +++ b/kubespray/roles/moaroom/templates/deploy-control.yml.j2 @@ -0,0 +1,102 @@ +apiVersion: v1 +kind: Namespace +metadata: + name: control-ns +--- +apiVersion: v1 +kind: Namespace +metadata: + name: professor-ns +--- +apiVersion: v1 +kind: Namespace +metadata: + name: student-ns +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: access-to-apiserver-sa + namespace: control-ns +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: access-to-apiserver-cr + namespace: control-ns +rules: + - apiGroups: + - "*" + resources: + - pods + - services + - nodes + verbs: + - "*" +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: access-to-apiserver-crb +subjects: + - kind: ServiceAccount + name: access-to-apiserver-sa + namespace: control-ns +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: access-to-apiserver-cr +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + name: control-ubuntu + namespace: control-ns + labels: + app: control-label +spec: + replicas: 1 + selector: + matchLabels: + app: control-label + template: + metadata: + labels: + app: control-label + spec: + containers: + - name: control-ubuntu + image: ryann3/control-ubuntu:5.0 + imagePullPolicy: Always + ports: + - name: nginx-port + containerPort: 80 + - name: ssh-port + containerPort: 22 + - name: server-port + containerPort: 8003 + - name: webssh-port + containerPort: 8886 + resources: {} + serviceAccountName: access-to-apiserver-sa +--- +apiVersion: v1 +kind: Service +metadata: + name: control-ubuntu-svc + namespace: control-ns +spec: + ports: + - name: http + port: 81 + targetPort: nginx-port + - name: webssh + port: 8886 + nodePort: 30000 + - name: server + port: 8003 + targetPort: server-port + nodePort: 30001 + selector: + app: control-label + type: NodePort diff --git a/kubespray/roles/network_plugin/calico/defaults/main.yml b/kubespray/roles/network_plugin/calico/defaults/main.yml new file mode 100644 index 0000000..b3c5f80 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/defaults/main.yml @@ -0,0 +1,171 @@ +--- +# the default value of name +calico_cni_name: k8s-pod-network + +# Enables Internet connectivity from containers +nat_outgoing: true + +# add default ippool name +calico_pool_name: "default-pool" +calico_ipv4pool_ipip: "Off" + +# Change encapsulation mode, by default we enable vxlan which is the most mature and well tested mode +calico_ipip_mode: Never # valid values are 'Always', 'Never' and 'CrossSubnet' +calico_vxlan_mode: Always # valid values are 'Always', 'Never' and 'CrossSubnet' + +calico_cni_pool: true +calico_cni_pool_ipv6: true + +# add default ippool blockSize (defaults kube_network_node_prefix) +calico_pool_blocksize: 26 + +# Calico doesn't support ipip tunneling for the IPv6. +calico_ipip_mode_ipv6: Never +calico_vxlan_mode_ipv6: Never + +# add default ipv6 ippool blockSize (defaults kube_network_node_prefix_ipv6) +calico_pool_blocksize_ipv6: 122 + +# Calico network backend can be 'bird', 'vxlan' and 'none' +calico_network_backend: vxlan + +calico_cert_dir: /etc/calico/certs + +# Global as_num (/calico/bgp/v1/global/as_num) +global_as_num: "64512" + +# You can set MTU value here. If left undefined or empty, it will +# not be specified in calico CNI config, so Calico will use built-in +# defaults. The value should be a number, not a string. +# calico_mtu: 1500 + +# Advertise Service External IPs +calico_advertise_service_external_ips: [] + +# Advertise Service LoadBalancer IPs +calico_advertise_service_loadbalancer_ips: [] + +# Calico eBPF support +calico_bpf_enabled: false +calico_bpf_log_level: "" +# Valid option for service mode: Tunnel (default), DSR=Direct Server Return +calico_bpf_service_mode: Tunnel + +# Calico floatingIPs support +# Valid option for floatingIPs: Disabled (default), Enabled +calico_felix_floating_ips: Disabled + +# Limits for apps +calico_node_memory_limit: 500M +calico_node_cpu_limit: 300m +calico_node_memory_requests: 64M +calico_node_cpu_requests: 150m +calico_felix_chaininsertmode: Insert + +# Calico daemonset nodeselector +calico_ds_nodeselector: "kubernetes.io/os: linux" + +# Virtual network ID to use for VXLAN traffic. A value of 0 means “use the kernel default”. +calico_vxlan_vni: 4096 + +# Port to use for VXLAN traffic. A value of 0 means “use the kernel default”. +calico_vxlan_port: 4789 + +# Enable Prometheus Metrics endpoint for felix +calico_felix_prometheusmetricsenabled: false +calico_felix_prometheusmetricsport: 9091 +calico_felix_prometheusgometricsenabled: true +calico_felix_prometheusprocessmetricsenabled: true + +# Set the agent log level. Can be debug, warning, info or fatal +calico_loglevel: info +calico_node_startup_loglevel: error + +# Set log path for calico CNI plugin. Set to false to disable logging to disk. +calico_cni_log_file_path: /var/log/calico/cni/cni.log + +# Enable or disable usage report to 'usage.projectcalico.org' +calico_usage_reporting: false + +# Should calico ignore kernel's RPF check setting, +# see https://github.com/projectcalico/felix/blob/ab8799eaea66627e5db7717e62fca61fd9c08646/python/calico/felix/config.py#L198 +calico_node_ignorelooserpf: false + +# Define address on which Felix will respond to health requests +calico_healthhost: "localhost" + +# Configure time in seconds that calico will wait for the iptables lock +calico_iptables_lock_timeout_secs: 10 + +# Choose Calico iptables backend: "Legacy", "Auto" or "NFT" (FELIX_IPTABLESBACKEND) +calico_iptables_backend: "Auto" + +# Calico Wireguard support +calico_wireguard_enabled: false +calico_wireguard_packages: [] +calico_wireguard_repo: https://download.copr.fedorainfracloud.org/results/jdoss/wireguard/epel-{{ ansible_distribution_major_version }}-$basearch/ + +# If you want to use non default IP_AUTODETECTION_METHOD, IP6_AUTODETECTION_METHOD for calico node set this option to one of: +# * can-reach=DESTINATION +# * interface=INTERFACE-REGEX +# see https://projectcalico.docs.tigera.io/reference/node/configuration#ip-autodetection-methods +# calico_ip_auto_method: "interface=eth.*" +# calico_ip6_auto_method: "interface=eth.*" + +# Set FELIX_MTUIFACEPATTERN, Pattern used to discover the host’s interface for MTU auto-detection. +# see https://projectcalico.docs.tigera.io/reference/felix/configuration +# calico_felix_mtu_iface_pattern: "^((en|wl|ww|sl|ib)[opsx].*|(eth|wlan|wwan).*)" + +calico_baremetal_nodename: "{{ kube_override_hostname | default(inventory_hostname) }}" + +kube_etcd_cacert_file: ca.pem +kube_etcd_cert_file: node-{{ inventory_hostname }}.pem +kube_etcd_key_file: node-{{ inventory_hostname }}-key.pem + +# Choose data store type for calico: "etcd" or "kdd" (kubernetes datastore) +# The default value for calico_datastore is set in role kubespray-default + +# Use typha (only with kdd) +typha_enabled: false +typha_prometheusmetricsenabled: false +typha_prometheusmetricsport: 9093 + +# Scaling typha: 1 replica per 100 nodes is adequate +# Number of typha replicas +typha_replicas: 1 + +# Set max typha connections +typha_max_connections_lower_limit: 300 + +# Generate certifcates for typha<->calico-node communication +typha_secure: false + +calico_feature_control: {} + +# Calico default BGP port +calico_bgp_listen_port: 179 + +# Calico FelixConfiguration options +calico_felix_reporting_interval: 0s +calico_felix_log_severity_screen: Info + +# Calico container settings +calico_allow_ip_forwarding: false + +# Calico IPAM strictAffinity +calico_ipam_strictaffinity: false + +# Calico IPAM autoAllocateBlocks +calico_ipam_autoallocateblocks: true + +# Calico IPAM maxBlocksPerHost, default 0 +calico_ipam_maxblocksperhost: 0 + +# Calico apiserver (only with kdd) +calico_apiserver_enabled: false + +# Calico feature detect override +calico_feature_detect_override: "" + +# Calico kubeconfig wait timeout in seconds +calico_kubeconfig_wait_timeout: 300 diff --git a/kubespray/roles/network_plugin/calico/files/openssl.conf b/kubespray/roles/network_plugin/calico/files/openssl.conf new file mode 100644 index 0000000..f4ba47d --- /dev/null +++ b/kubespray/roles/network_plugin/calico/files/openssl.conf @@ -0,0 +1,27 @@ +req_extensions = v3_req +distinguished_name = req_distinguished_name + +[req_distinguished_name] + +[ v3_req ] +basicConstraints = CA:FALSE +keyUsage = digitalSignature, keyEncipherment + +[ ssl_client ] +extendedKeyUsage = clientAuth, serverAuth +basicConstraints = CA:FALSE +subjectKeyIdentifier=hash +authorityKeyIdentifier=keyid,issuer + +[ v3_ca ] +basicConstraints = CA:TRUE +keyUsage = cRLSign, digitalSignature, keyCertSign +subjectKeyIdentifier=hash +authorityKeyIdentifier=keyid:always,issuer + +[ ssl_client_apiserver ] +extendedKeyUsage = clientAuth, serverAuth +basicConstraints = CA:FALSE +subjectKeyIdentifier=hash +authorityKeyIdentifier=keyid,issuer +subjectAltName = DNS:calico-api.calico-apiserver.svc diff --git a/kubespray/roles/network_plugin/calico/handlers/main.yml b/kubespray/roles/network_plugin/calico/handlers/main.yml new file mode 100644 index 0000000..7f998db --- /dev/null +++ b/kubespray/roles/network_plugin/calico/handlers/main.yml @@ -0,0 +1,31 @@ +--- +- name: Reset_calico_cni + command: /bin/true + when: calico_cni_config is defined + notify: + - Delete 10-calico.conflist + - Calico | delete calico-node docker containers + - Calico | delete calico-node crio/containerd containers + +- name: Delete 10-calico.conflist + file: + path: /etc/cni/net.d/10-calico.conflist + state: absent + +- name: Calico | delete calico-node docker containers + shell: "set -o pipefail && {{ docker_bin_dir }}/docker ps -af name=k8s_POD_calico-node* -q | xargs --no-run-if-empty {{ docker_bin_dir }}/docker rm -f" + args: + executable: /bin/bash + register: docker_calico_node_remove + until: docker_calico_node_remove is succeeded + retries: 5 + when: container_manager in ["docker"] + +- name: Calico | delete calico-node crio/containerd containers + shell: 'set -o pipefail && {{ bin_dir }}/crictl pods --name calico-node-* -q | xargs -I% --no-run-if-empty bash -c "{{ bin_dir }}/crictl stopp % && {{ bin_dir }}/crictl rmp %"' + args: + executable: /bin/bash + register: crictl_calico_node_remove + until: crictl_calico_node_remove is succeeded + retries: 5 + when: container_manager in ["crio", "containerd"] diff --git a/kubespray/roles/network_plugin/calico/rr/defaults/main.yml b/kubespray/roles/network_plugin/calico/rr/defaults/main.yml new file mode 100644 index 0000000..dedda19 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/rr/defaults/main.yml @@ -0,0 +1,5 @@ +--- +# Global as_num (/calico/bgp/v1/global/as_num) +# should be the same as in calico role +global_as_num: "64512" +calico_baremetal_nodename: "{{ kube_override_hostname | default(inventory_hostname) }}" diff --git a/kubespray/roles/network_plugin/calico/rr/tasks/main.yml b/kubespray/roles/network_plugin/calico/rr/tasks/main.yml new file mode 100644 index 0000000..471518d --- /dev/null +++ b/kubespray/roles/network_plugin/calico/rr/tasks/main.yml @@ -0,0 +1,16 @@ +--- +- name: Calico-rr | Pre-upgrade tasks + include_tasks: pre.yml + +- name: Calico-rr | Configuring node tasks + include_tasks: update-node.yml + +- name: Calico-rr | Set label for route reflector + command: >- + {{ bin_dir }}/calicoctl.sh label node {{ inventory_hostname }} + 'i-am-a-route-reflector=true' --overwrite + changed_when: false + register: calico_rr_label + until: calico_rr_label is succeeded + delay: "{{ retry_stagger | random + 3 }}" + retries: 10 diff --git a/kubespray/roles/network_plugin/calico/rr/tasks/pre.yml b/kubespray/roles/network_plugin/calico/rr/tasks/pre.yml new file mode 100644 index 0000000..d8dbd80 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/rr/tasks/pre.yml @@ -0,0 +1,15 @@ +--- +- name: Calico-rr | Disable calico-rr service if it exists + service: + name: calico-rr + state: stopped + enabled: no + failed_when: false + +- name: Calico-rr | Delete obsolete files + file: + path: "{{ item }}" + state: absent + with_items: + - /etc/calico/calico-rr.env + - /etc/systemd/system/calico-rr.service diff --git a/kubespray/roles/network_plugin/calico/rr/tasks/update-node.yml b/kubespray/roles/network_plugin/calico/rr/tasks/update-node.yml new file mode 100644 index 0000000..fc873ba --- /dev/null +++ b/kubespray/roles/network_plugin/calico/rr/tasks/update-node.yml @@ -0,0 +1,50 @@ +--- +# Workaround to retry a block of tasks, ansible doesn't have a direct way to do it, +# you can follow the block loop request in: https://github.com/ansible/ansible/issues/46203 +- name: Calico-rr | Configure route reflector + block: + - name: Set the retry count + set_fact: + retry_count: "{{ 0 if retry_count is undefined else retry_count | int + 1 }}" + + - name: Calico | Set label for route reflector # noqa command-instead-of-shell + shell: "{{ bin_dir }}/calicoctl.sh label node {{ inventory_hostname }} calico-rr-id={{ calico_rr_id }} --overwrite" + changed_when: false + register: calico_rr_id_label + until: calico_rr_id_label is succeeded + delay: "{{ retry_stagger | random + 3 }}" + retries: 10 + when: calico_rr_id is defined + + - name: Calico-rr | Fetch current node object + command: "{{ bin_dir }}/calicoctl.sh get node {{ inventory_hostname }} -ojson" + changed_when: false + register: calico_rr_node + until: calico_rr_node is succeeded + delay: "{{ retry_stagger | random + 3 }}" + retries: 10 + + - name: Calico-rr | Set route reflector cluster ID + # noqa: jinja[spacing] + set_fact: + calico_rr_node_patched: >- + {{ calico_rr_node.stdout | from_json | combine({ 'spec': { 'bgp': + { 'routeReflectorClusterID': cluster_id }}}, recursive=True) }} + + - name: Calico-rr | Configure route reflector # noqa command-instead-of-shell + shell: "{{ bin_dir }}/calicoctl.sh replace -f-" + args: + stdin: "{{ calico_rr_node_patched | to_json }}" + + rescue: + - name: Fail if retry limit is reached + fail: + msg: Ended after 10 retries + when: retry_count | int == 10 + + - name: Retrying node configuration + debug: + msg: "Failed to configure route reflector - Retrying..." + + - name: Retry node configuration + include_tasks: update-node.yml diff --git a/kubespray/roles/network_plugin/calico/tasks/calico_apiserver_certs.yml b/kubespray/roles/network_plugin/calico/tasks/calico_apiserver_certs.yml new file mode 100644 index 0000000..fc336e4 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/tasks/calico_apiserver_certs.yml @@ -0,0 +1,60 @@ +--- +- name: Calico | Check if calico apiserver exists + command: "{{ kubectl }} -n calico-apiserver get secret calico-apiserver-certs" + register: calico_apiserver_secret + changed_when: false + failed_when: false + +- name: Calico | Create ns manifests + template: + src: "calico-apiserver-ns.yml.j2" + dest: "{{ kube_config_dir }}/calico-apiserver-ns.yml" + mode: 0644 + +- name: Calico | Apply ns manifests + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/calico-apiserver-ns.yml" + state: "latest" + +- name: Calico | Ensure calico certs dir + file: + path: /etc/calico/certs + state: directory + mode: 0755 + when: calico_apiserver_secret.rc != 0 + +- name: Calico | Copy ssl script for apiserver certs + template: + src: make-ssl-calico.sh.j2 + dest: "{{ bin_dir }}/make-ssl-apiserver.sh" + mode: 0755 + when: calico_apiserver_secret.rc != 0 + +- name: Calico | Copy ssl config for apiserver certs + copy: + src: openssl.conf + dest: /etc/calico/certs/openssl.conf + mode: 0644 + when: calico_apiserver_secret.rc != 0 + +- name: Calico | Generate apiserver certs + command: >- + {{ bin_dir }}/make-ssl-apiserver.sh + -f /etc/calico/certs/openssl.conf + -c {{ kube_cert_dir }} + -d /etc/calico/certs + -s apiserver + when: calico_apiserver_secret.rc != 0 + +- name: Calico | Create calico apiserver generic secrets + command: >- + {{ kubectl }} -n calico-apiserver + create secret generic {{ item.name }} + --from-file={{ item.cert }} + --from-file={{ item.key }} + with_items: + - name: calico-apiserver-certs + cert: /etc/calico/certs/apiserver.crt + key: /etc/calico/certs/apiserver.key + when: calico_apiserver_secret.rc != 0 diff --git a/kubespray/roles/network_plugin/calico/tasks/check.yml b/kubespray/roles/network_plugin/calico/tasks/check.yml new file mode 100644 index 0000000..2138be8 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/tasks/check.yml @@ -0,0 +1,207 @@ +--- +- name: Stop if legacy encapsulation variables are detected (ipip) + assert: + that: + - ipip is not defined + msg: "'ipip' configuration variable is deprecated, please configure your inventory with 'calico_ipip_mode' set to 'Always' or 'CrossSubnet' according to your specific needs" + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: Stop if legacy encapsulation variables are detected (ipip_mode) + assert: + that: + - ipip_mode is not defined + msg: "'ipip_mode' configuration variable is deprecated, please configure your inventory with 'calico_ipip_mode' set to 'Always' or 'CrossSubnet' according to your specific needs" + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: Stop if legacy encapsulation variables are detected (calcio_ipam_autoallocateblocks) + assert: + that: + - calcio_ipam_autoallocateblocks is not defined + msg: "'calcio_ipam_autoallocateblocks' configuration variable is deprecated, it's a typo, please configure your inventory with 'calico_ipam_autoallocateblocks' set to 'true' or 'false' according to your specific needs" + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + + +- name: Stop if incompatible network plugin and cloudprovider + assert: + that: + - calico_ipip_mode == 'Never' + - calico_vxlan_mode in ['Always', 'CrossSubnet'] + msg: "When using cloud_provider azure and network_plugin calico calico_ipip_mode must be 'Never' and calico_vxlan_mode 'Always' or 'CrossSubnet'" + when: + - cloud_provider is defined and cloud_provider == 'azure' + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: Stop if supported Calico versions + assert: + that: + - "calico_version in calico_crds_archive_checksums.keys()" + msg: "Calico version not supported {{ calico_version }} not in {{ calico_crds_archive_checksums.keys() }}" + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: Check if calicoctl.sh exists + stat: + path: "{{ bin_dir }}/calicoctl.sh" + register: calicoctl_sh_exists + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: Check if calico ready + command: "{{ bin_dir }}/calicoctl.sh get ClusterInformation default" + register: calico_ready + run_once: True + ignore_errors: True + retries: 5 + delay: 10 + until: calico_ready.rc == 0 + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: calicoctl_sh_exists.stat.exists + +- name: Check that current calico version is enough for upgrade + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: calicoctl_sh_exists.stat.exists and calico_ready.rc == 0 + block: + - name: Get current calico version + shell: "set -o pipefail && {{ bin_dir }}/calicoctl.sh version | grep 'Client Version:' | awk '{ print $3}'" + args: + executable: /bin/bash + register: calico_version_on_server + changed_when: false + + - name: Assert that current calico version is enough for upgrade + assert: + that: + - calico_version_on_server.stdout is version(calico_min_version_required, '>=') + msg: > + Your version of calico is not fresh enough for upgrade. + Minimum version is {{ calico_min_version_required }} supported by the previous kubespray release. + But current version is {{ calico_version_on_server.stdout }}. + +- name: "Check that cluster_id is set if calico_rr enabled" + assert: + that: + - cluster_id is defined + msg: "A unique cluster_id is required if using calico_rr" + when: + - peer_with_calico_rr + - inventory_hostname == groups['kube_control_plane'][0] + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: "Check that calico_rr nodes are in k8s_cluster group" + assert: + that: + - '"k8s_cluster" in group_names' + msg: "calico_rr must be a child group of k8s_cluster group" + when: + - '"calico_rr" in group_names' + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: "Check vars defined correctly" + assert: + that: + - "calico_pool_name is defined" + - "calico_pool_name is match('^[a-zA-Z0-9-_\\\\.]{2,63}$')" + msg: "calico_pool_name contains invalid characters" + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: "Check calico network backend defined correctly" + assert: + that: + - "calico_network_backend in ['bird', 'vxlan', 'none']" + msg: "calico network backend is not 'bird', 'vxlan' or 'none'" + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: "Check ipip and vxlan mode defined correctly" + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + assert: + that: + - "calico_ipip_mode in ['Always', 'CrossSubnet', 'Never']" + - "calico_vxlan_mode in ['Always', 'CrossSubnet', 'Never']" + msg: "calico inter host encapsulation mode is not 'Always', 'CrossSubnet' or 'Never'" + +- name: "Check ipip and vxlan mode if simultaneously enabled" + assert: + that: + - "calico_vxlan_mode in ['Never']" + msg: "IP in IP and VXLAN mode is mutualy exclusive modes" + when: + - "calico_ipip_mode in ['Always', 'CrossSubnet']" + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: "Check ipip and vxlan mode if simultaneously enabled" + assert: + that: + - "calico_ipip_mode in ['Never']" + msg: "IP in IP and VXLAN mode is mutualy exclusive modes" + when: + - "calico_vxlan_mode in ['Always', 'CrossSubnet']" + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: "Get Calico {{ calico_pool_name }} configuration" + command: "{{ bin_dir }}/calicoctl.sh get ipPool {{ calico_pool_name }} -o json" + failed_when: False + changed_when: False + check_mode: no + register: calico + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: "Set calico_pool_conf" + set_fact: + calico_pool_conf: '{{ calico.stdout | from_json }}' + when: calico.rc == 0 and calico.stdout + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: "Check if inventory match current cluster configuration" + assert: + that: + - calico_pool_conf.spec.blockSize | int == (calico_pool_blocksize | default(kube_network_node_prefix) | int) + - calico_pool_conf.spec.cidr == (calico_pool_cidr | default(kube_pods_subnet)) + - not calico_pool_conf.spec.ipipMode is defined or calico_pool_conf.spec.ipipMode == calico_ipip_mode + - not calico_pool_conf.spec.vxlanMode is defined or calico_pool_conf.spec.vxlanMode == calico_vxlan_mode + msg: "Your inventory doesn't match the current cluster configuration" + when: + - calico_pool_conf is defined + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: "Check kdd calico_datastore if calico_apiserver_enabled" + assert: + that: calico_datastore == "kdd" + msg: "When using calico apiserver you need to use the kubernetes datastore" + when: + - calico_apiserver_enabled + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: "Check kdd calico_datastore if typha_enabled" + assert: + that: calico_datastore == "kdd" + msg: "When using typha you need to use the kubernetes datastore" + when: + - typha_enabled + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: "Check ipip mode is Never for calico ipv6" + assert: + that: + - "calico_ipip_mode_ipv6 in ['Never']" + msg: "Calico doesn't support ipip tunneling for the IPv6" + when: + - enable_dual_stack_networks + run_once: True + delegate_to: "{{ groups['kube_control_plane'][0] }}" diff --git a/kubespray/roles/network_plugin/calico/tasks/install.yml b/kubespray/roles/network_plugin/calico/tasks/install.yml new file mode 100644 index 0000000..d371440 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/tasks/install.yml @@ -0,0 +1,479 @@ +--- +- name: Calico | Install Wireguard packages + package: + name: "{{ item }}" + state: present + with_items: "{{ calico_wireguard_packages }}" + register: calico_package_install + until: calico_package_install is succeeded + retries: 4 + when: calico_wireguard_enabled + +- name: Calico | Copy calicoctl binary from download dir + copy: + src: "{{ downloads.calicoctl.dest }}" + dest: "{{ bin_dir }}/calicoctl" + mode: 0755 + remote_src: yes + +- name: Calico | Create calico certs directory + file: + dest: "{{ calico_cert_dir }}" + state: directory + mode: 0750 + owner: root + group: root + when: calico_datastore == "etcd" + +- name: Calico | Link etcd certificates for calico-node + file: + src: "{{ etcd_cert_dir }}/{{ item.s }}" + dest: "{{ calico_cert_dir }}/{{ item.d }}" + state: hard + mode: 0640 + force: yes + with_items: + - {s: "{{ kube_etcd_cacert_file }}", d: "ca_cert.crt"} + - {s: "{{ kube_etcd_cert_file }}", d: "cert.crt"} + - {s: "{{ kube_etcd_key_file }}", d: "key.pem"} + when: calico_datastore == "etcd" + +- name: Calico | Generate typha certs + include_tasks: typha_certs.yml + when: + - typha_secure + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Calico | Generate apiserver certs + include_tasks: calico_apiserver_certs.yml + when: + - calico_apiserver_enabled + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Calico | Install calicoctl wrapper script + template: + src: "calicoctl.{{ calico_datastore }}.sh.j2" + dest: "{{ bin_dir }}/calicoctl.sh" + mode: 0755 + owner: root + group: root + +- name: Calico | wait for etcd + uri: + url: "{{ etcd_access_addresses.split(',') | first }}/health" + validate_certs: no + client_cert: "{{ calico_cert_dir }}/cert.crt" + client_key: "{{ calico_cert_dir }}/key.pem" + register: result + until: result.status == 200 or result.status == 401 + retries: 10 + delay: 5 + run_once: true + when: calico_datastore == "etcd" + +- name: Calico | Check if calico network pool has already been configured + # noqa risky-shell-pipe - grep will exit 1 if no match found + shell: > + {{ bin_dir }}/calicoctl.sh get ippool | grep -w "{{ calico_pool_cidr | default(kube_pods_subnet) }}" | wc -l + args: + executable: /bin/bash + register: calico_conf + retries: 4 + until: calico_conf.rc == 0 + delay: "{{ retry_stagger | random + 3 }}" + changed_when: false + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Calico | Ensure that calico_pool_cidr is within kube_pods_subnet when defined + assert: + that: "[calico_pool_cidr] | ipaddr(kube_pods_subnet) | length == 1" + msg: "{{ calico_pool_cidr }} is not within or equal to {{ kube_pods_subnet }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - 'calico_conf.stdout == "0"' + - calico_pool_cidr is defined + +- name: Calico | Check if calico IPv6 network pool has already been configured + # noqa risky-shell-pipe - grep will exit 1 if no match found + shell: > + {{ bin_dir }}/calicoctl.sh get ippool | grep -w "{{ calico_pool_cidr_ipv6 | default(kube_pods_subnet_ipv6) }}" | wc -l + args: + executable: /bin/bash + register: calico_conf_ipv6 + retries: 4 + until: calico_conf_ipv6.rc == 0 + delay: "{{ retry_stagger | random + 3 }}" + changed_when: false + when: + - inventory_hostname == groups['kube_control_plane'][0] + - enable_dual_stack_networks + +- name: Calico | Ensure that calico_pool_cidr_ipv6 is within kube_pods_subnet_ipv6 when defined + assert: + that: "[calico_pool_cidr_ipv6] | ipaddr(kube_pods_subnet_ipv6) | length == 1" + msg: "{{ calico_pool_cidr_ipv6 }} is not within or equal to {{ kube_pods_subnet_ipv6 }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - calico_conf_ipv6.stdout is defined and calico_conf_ipv6.stdout == "0" + - calico_pool_cidr_ipv6 is defined + - enable_dual_stack_networks + +- name: Calico | kdd specific configuration + when: + - inventory_hostname in groups['kube_control_plane'] + - calico_datastore == "kdd" + block: + - name: Calico | Check if extra directory is needed + stat: + path: "{{ local_release_dir }}/calico-{{ calico_version }}-kdd-crds/{{ 'kdd' if (calico_version is version('v3.22.3', '<')) else 'crd' }}" + register: kdd_path + - name: Calico | Set kdd path when calico < v3.22.3 + set_fact: + calico_kdd_path: "{{ local_release_dir }}/calico-{{ calico_version }}-kdd-crds{{ '/kdd' if kdd_path.stat.exists is defined and kdd_path.stat.exists }}" + when: + - calico_version is version('v3.22.3', '<') + - name: Calico | Set kdd path when calico > v3.22.2 + set_fact: + calico_kdd_path: "{{ local_release_dir }}/calico-{{ calico_version }}-kdd-crds{{ '/crd' if kdd_path.stat.exists is defined and kdd_path.stat.exists }}" + when: + - calico_version is version('v3.22.2', '>') + - name: Calico | Create calico manifests for kdd + assemble: + src: "{{ calico_kdd_path }}" + dest: "{{ kube_config_dir }}/kdd-crds.yml" + mode: 0644 + delimiter: "---\n" + regexp: ".*\\.yaml" + remote_src: true + + - name: Calico | Create Calico Kubernetes datastore resources + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/kdd-crds.yml" + state: "latest" + register: kubectl_result + until: kubectl_result is succeeded + retries: 5 + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Calico | Configure Felix + when: + - inventory_hostname == groups['kube_control_plane'][0] + block: + - name: Calico | Get existing FelixConfiguration + command: "{{ bin_dir }}/calicoctl.sh get felixconfig default -o json" + register: _felix_cmd + ignore_errors: True + changed_when: False + + - name: Calico | Set kubespray FelixConfiguration + set_fact: + _felix_config: > + { + "kind": "FelixConfiguration", + "apiVersion": "projectcalico.org/v3", + "metadata": { + "name": "default", + }, + "spec": { + "ipipEnabled": {{ calico_ipip_mode != 'Never' }}, + "reportingInterval": "{{ calico_felix_reporting_interval }}", + "bpfLogLevel": "{{ calico_bpf_log_level }}", + "bpfEnabled": {{ calico_bpf_enabled | bool }}, + "bpfExternalServiceMode": "{{ calico_bpf_service_mode }}", + "wireguardEnabled": {{ calico_wireguard_enabled | bool }}, + "logSeverityScreen": "{{ calico_felix_log_severity_screen }}", + "vxlanEnabled": {{ calico_vxlan_mode != 'Never' }}, + "featureDetectOverride": "{{ calico_feature_detect_override }}", + "floatingIPs": "{{ calico_felix_floating_ips }}" + } + } + + - name: Calico | Process FelixConfiguration + set_fact: + _felix_config: "{{ _felix_cmd.stdout | from_json | combine(_felix_config, recursive=True) }}" + when: + - _felix_cmd is success + + - name: Calico | Configure calico FelixConfiguration + command: + cmd: "{{ bin_dir }}/calicoctl.sh apply -f -" + stdin: "{{ _felix_config is string | ternary(_felix_config, _felix_config | to_json) }}" + changed_when: False + +- name: Calico | Configure Calico IP Pool + when: + - inventory_hostname == groups['kube_control_plane'][0] + block: + - name: Calico | Get existing calico network pool + command: "{{ bin_dir }}/calicoctl.sh get ippool {{ calico_pool_name }} -o json" + register: _calico_pool_cmd + ignore_errors: True + changed_when: False + + - name: Calico | Set kubespray calico network pool + set_fact: + _calico_pool: > + { + "kind": "IPPool", + "apiVersion": "projectcalico.org/v3", + "metadata": { + "name": "{{ calico_pool_name }}", + }, + "spec": { + "blockSize": {{ calico_pool_blocksize | default(kube_network_node_prefix) }}, + "cidr": "{{ calico_pool_cidr | default(kube_pods_subnet) }}", + "ipipMode": "{{ calico_ipip_mode }}", + "vxlanMode": "{{ calico_vxlan_mode }}", + "natOutgoing": {{ nat_outgoing | default(false) }} + } + } + + - name: Calico | Process calico network pool + set_fact: + _calico_pool: "{{ _calico_pool_cmd.stdout | from_json | combine(_calico_pool, recursive=True) }}" + when: + - _calico_pool_cmd is success + + - name: Calico | Configure calico network pool + command: + cmd: "{{ bin_dir }}/calicoctl.sh apply -f -" + stdin: "{{ _calico_pool is string | ternary(_calico_pool, _calico_pool | to_json) }}" + changed_when: False + +- name: Calico | Configure Calico IPv6 Pool + when: + - inventory_hostname == groups['kube_control_plane'][0] + - enable_dual_stack_networks | bool + block: + - name: Calico | Get existing calico ipv6 network pool + command: "{{ bin_dir }}/calicoctl.sh get ippool {{ calico_pool_name }}-ipv6 -o json" + register: _calico_pool_ipv6_cmd + ignore_errors: True + changed_when: False + + - name: Calico | Set kubespray calico network pool + set_fact: + _calico_pool_ipv6: > + { + "kind": "IPPool", + "apiVersion": "projectcalico.org/v3", + "metadata": { + "name": "{{ calico_pool_name }}-ipv6", + }, + "spec": { + "blockSize": {{ calico_pool_blocksize_ipv6 | default(kube_network_node_prefix_ipv6) }}, + "cidr": "{{ calico_pool_cidr_ipv6 | default(kube_pods_subnet_ipv6) }}", + "ipipMode": "{{ calico_ipip_mode_ipv6 }}", + "vxlanMode": "{{ calico_vxlan_mode_ipv6 }}", + "natOutgoing": {{ nat_outgoing_ipv6 | default(false) }} + } + } + + - name: Calico | Process calico ipv6 network pool + set_fact: + _calico_pool_ipv6: "{{ _calico_pool_ipv6_cmd.stdout | from_json | combine(_calico_pool_ipv6, recursive=True) }}" + when: + - _calico_pool_ipv6_cmd is success + + - name: Calico | Configure calico ipv6 network pool + command: + cmd: "{{ bin_dir }}/calicoctl.sh apply -f -" + stdin: "{{ _calico_pool_ipv6 is string | ternary(_calico_pool_ipv6, _calico_pool_ipv6 | to_json) }}" + changed_when: False + +- name: Populate Service External IPs + set_fact: + _service_external_ips: "{{ _service_external_ips | default([]) + [{'cidr': item}] }}" + with_items: "{{ calico_advertise_service_external_ips }}" + run_once: yes + +- name: Populate Service LoadBalancer IPs + set_fact: + _service_loadbalancer_ips: "{{ _service_loadbalancer_ips | default([]) + [{'cidr': item}] }}" + with_items: "{{ calico_advertise_service_loadbalancer_ips }}" + run_once: yes + +- name: "Determine nodeToNodeMesh needed state" + set_fact: + nodeToNodeMeshEnabled: "false" + when: + - peer_with_router | default(false) or peer_with_calico_rr | default(false) + - inventory_hostname in groups['k8s_cluster'] + run_once: yes + +- name: Calico | Configure Calico BGP + when: + - inventory_hostname == groups['kube_control_plane'][0] + block: + - name: Calico | Get existing BGP Configuration + command: "{{ bin_dir }}/calicoctl.sh get bgpconfig default -o json" + register: _bgp_config_cmd + ignore_errors: True + changed_when: False + + - name: Calico | Set kubespray BGP Configuration + set_fact: + # noqa: jinja[spacing] + _bgp_config: > + { + "kind": "BGPConfiguration", + "apiVersion": "projectcalico.org/v3", + "metadata": { + "name": "default", + }, + "spec": { + "listenPort": {{ calico_bgp_listen_port }}, + "logSeverityScreen": "Info", + {% if not calico_no_global_as_num | default(false) %}"asNumber": {{ global_as_num }},{% endif %} + "nodeToNodeMeshEnabled": {{ nodeToNodeMeshEnabled | default('true') }} , + {% if calico_advertise_cluster_ips | default(false) %} + "serviceClusterIPs": [{"cidr": "{{ kube_service_addresses }}" } {{ ',{"cidr":"' + kube_service_addresses_ipv6 + '"}' if enable_dual_stack_networks else '' }}],{% endif %} + {% if calico_advertise_service_loadbalancer_ips | length > 0 %}"serviceLoadBalancerIPs": {{ _service_loadbalancer_ips }},{% endif %} + "serviceExternalIPs": {{ _service_external_ips | default([]) }} + } + } + + - name: Calico | Process BGP Configuration + set_fact: + _bgp_config: "{{ _bgp_config_cmd.stdout | from_json | combine(_bgp_config, recursive=True) }}" + when: + - _bgp_config_cmd is success + + - name: Calico | Set up BGP Configuration + command: + cmd: "{{ bin_dir }}/calicoctl.sh apply -f -" + stdin: "{{ _bgp_config is string | ternary(_bgp_config, _bgp_config | to_json) }}" + changed_when: False + +- name: Calico | Create calico manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: calico-config, file: calico-config.yml, type: cm} + - {name: calico-node, file: calico-node.yml, type: ds} + - {name: calico, file: calico-node-sa.yml, type: sa} + - {name: calico, file: calico-cr.yml, type: clusterrole} + - {name: calico, file: calico-crb.yml, type: clusterrolebinding} + - {name: kubernetes-services-endpoint, file: kubernetes-services-endpoint.yml, type: cm } + register: calico_node_manifests + when: + - inventory_hostname in groups['kube_control_plane'] + - rbac_enabled or item.type not in rbac_resources + +- name: Calico | Create calico manifests for typha + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: calico, file: calico-typha.yml, type: typha} + register: calico_node_typha_manifest + when: + - inventory_hostname in groups['kube_control_plane'] + - typha_enabled + +- name: Calico | get calico apiserver caBundle + command: "{{ bin_dir }}/kubectl get secret -n calico-apiserver calico-apiserver-certs -o jsonpath='{.data.apiserver\\.crt}'" + changed_when: false + register: calico_apiserver_cabundle + when: + - inventory_hostname == groups['kube_control_plane'][0] + - calico_apiserver_enabled + +- name: Calico | set calico apiserver caBundle fact + set_fact: + calico_apiserver_cabundle: "{{ calico_apiserver_cabundle.stdout }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - calico_apiserver_enabled + +- name: Calico | Create calico manifests for apiserver + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: calico, file: calico-apiserver.yml, type: calico-apiserver} + register: calico_apiserver_manifest + when: + - inventory_hostname in groups['kube_control_plane'] + - calico_apiserver_enabled + +- name: Start Calico resources + kube: + name: "{{ item.item.name }}" + namespace: "kube-system" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ calico_node_manifests.results }}" + - "{{ calico_node_typha_manifest.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item.file }}" + +- name: Start Calico apiserver resources + kube: + name: "{{ item.item.name }}" + namespace: "calico-apiserver" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/{{ item.item.file }}" + state: "latest" + with_items: + - "{{ calico_apiserver_manifest.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not item is skipped + loop_control: + label: "{{ item.item.file }}" + +- name: Wait for calico kubeconfig to be created + wait_for: + path: /etc/cni/net.d/calico-kubeconfig + timeout: "{{ calico_kubeconfig_wait_timeout }}" + when: + - inventory_hostname not in groups['kube_control_plane'] + - calico_datastore == "kdd" + +- name: Calico | Create Calico ipam manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: calico, file: calico-ipamconfig.yml, type: ipam} + when: + - inventory_hostname in groups['kube_control_plane'] + - calico_datastore == "kdd" + +- name: Calico | Create ipamconfig resources + kube: + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/calico-ipamconfig.yml" + state: "latest" + register: resource_result + until: resource_result is succeeded + retries: 4 + when: + - inventory_hostname == groups['kube_control_plane'][0] + - calico_datastore == "kdd" + +- name: Calico | Peer with Calico Route Reflector + include_tasks: peer_with_calico_rr.yml + when: + - peer_with_calico_rr | default(false) + +- name: Calico | Peer with the router + include_tasks: peer_with_router.yml + when: + - peer_with_router | default(false) diff --git a/kubespray/roles/network_plugin/calico/tasks/main.yml b/kubespray/roles/network_plugin/calico/tasks/main.yml new file mode 100644 index 0000000..5921a91 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/tasks/main.yml @@ -0,0 +1,9 @@ +--- +- name: Calico Pre tasks + import_tasks: pre.yml + +- name: Calico repos + import_tasks: repos.yml + +- name: Calico install + include_tasks: install.yml diff --git a/kubespray/roles/network_plugin/calico/tasks/peer_with_calico_rr.yml b/kubespray/roles/network_plugin/calico/tasks/peer_with_calico_rr.yml new file mode 100644 index 0000000..9d216bd --- /dev/null +++ b/kubespray/roles/network_plugin/calico/tasks/peer_with_calico_rr.yml @@ -0,0 +1,86 @@ +--- +- name: Calico | Set label for groups nodes + command: "{{ bin_dir }}/calicoctl.sh label node {{ inventory_hostname }} calico-group-id={{ calico_group_id }} --overwrite" + changed_when: false + register: calico_group_id_label + until: calico_group_id_label is succeeded + delay: "{{ retry_stagger | random + 3 }}" + retries: 10 + when: + - calico_group_id is defined + +- name: Calico | Configure peering with route reflectors at global scope + command: + cmd: "{{ bin_dir }}/calicoctl.sh apply -f -" + # revert when it's already a string + stdin: "{{ stdin is string | ternary(stdin, stdin | to_json) }}" + vars: + stdin: > + {"apiVersion": "projectcalico.org/v3", + "kind": "BGPPeer", + "metadata": { + "name": "{{ calico_rr_id }}-to-node" + }, + "spec": { + "peerSelector": "calico-rr-id == '{{ calico_rr_id }}'", + "nodeSelector": "calico-group-id == '{{ calico_group_id }}'" + }} + register: output + retries: 4 + until: output.rc == 0 + delay: "{{ retry_stagger | random + 3 }}" + when: + - calico_rr_id is defined + - calico_group_id is defined + - inventory_hostname in groups['calico_rr'] + +- name: Calico | Configure peering with route reflectors at global scope + command: + cmd: "{{ bin_dir }}/calicoctl.sh apply -f -" + # revert when it's already a string + stdin: "{{ stdin is string | ternary(stdin, stdin | to_json) }}" + vars: + stdin: > + {"apiVersion": "projectcalico.org/v3", + "kind": "BGPPeer", + "metadata": { + "name": "peer-to-rrs" + }, + "spec": { + "nodeSelector": "!has(i-am-a-route-reflector)", + "peerSelector": "has(i-am-a-route-reflector)" + }} + register: output + retries: 4 + until: output.rc == 0 + delay: "{{ retry_stagger | random + 3 }}" + with_items: + - "{{ groups['calico_rr'] | default([]) }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - calico_rr_id is not defined or calico_group_id is not defined + +- name: Calico | Configure route reflectors to peer with each other + command: + cmd: "{{ bin_dir }}/calicoctl.sh apply -f -" + # revert when it's already a string + stdin: "{{ stdin is string | ternary(stdin, stdin | to_json) }}" + vars: + stdin: > + {"apiVersion": "projectcalico.org/v3", + "kind": "BGPPeer", + "metadata": { + "name": "rr-mesh" + }, + "spec": { + "nodeSelector": "has(i-am-a-route-reflector)", + "peerSelector": "has(i-am-a-route-reflector)" + }} + register: output + retries: 4 + until: output.rc == 0 + delay: "{{ retry_stagger | random + 3 }}" + with_items: + - "{{ groups['calico_rr'] | default([]) }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/network_plugin/calico/tasks/peer_with_router.yml b/kubespray/roles/network_plugin/calico/tasks/peer_with_router.yml new file mode 100644 index 0000000..a29ca36 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/tasks/peer_with_router.yml @@ -0,0 +1,77 @@ +--- +- name: Calico | Configure peering with router(s) at global scope + command: + cmd: "{{ bin_dir }}/calicoctl.sh apply -f -" + stdin: "{{ stdin is string | ternary(stdin, stdin | to_json) }}" + vars: + stdin: > + {"apiVersion": "projectcalico.org/v3", + "kind": "BGPPeer", + "metadata": { + "name": "global-{{ item.name | default(item.router_id | replace(':', '-')) }}" + }, + "spec": { + "asNumber": "{{ item.as }}", + "peerIP": "{{ item.router_id }}" + }} + register: output + retries: 4 + until: output.rc == 0 + delay: "{{ retry_stagger | random + 3 }}" + with_items: + - "{{ peers | selectattr('scope', 'defined') | selectattr('scope', 'equalto', 'global') | list | default([]) }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Calico | Configure node asNumber for per node peering + command: + cmd: "{{ bin_dir }}/calicoctl.sh apply -f -" + stdin: "{{ stdin is string | ternary(stdin, stdin | to_json) }}" + vars: + stdin: > + {"apiVersion": "projectcalico.org/v3", + "kind": "Node", + "metadata": { + "name": "{{ inventory_hostname }}" + }, + "spec": { + "bgp": { + "asNumber": "{{ local_as }}" + }, + "orchRefs":[{"nodeName":"{{ inventory_hostname }}","orchestrator":"k8s"}] + }} + register: output + retries: 4 + until: output.rc == 0 + delay: "{{ retry_stagger | random + 3 }}" + when: + - inventory_hostname in groups['k8s_cluster'] + - local_as is defined + - groups['calico_rr'] | default([]) | length == 0 + +- name: Calico | Configure peering with router(s) at node scope + command: + cmd: "{{ bin_dir }}/calicoctl.sh apply -f -" + stdin: "{{ stdin is string | ternary(stdin, stdin | to_json) }}" + vars: + stdin: > + {"apiVersion": "projectcalico.org/v3", + "kind": "BGPPeer", + "metadata": { + "name": "{{ inventory_hostname }}-{{ item.name | default(item.router_id | replace(':', '-')) }}" + }, + "spec": { + "asNumber": "{{ item.as }}", + "node": "{{ inventory_hostname }}", + "peerIP": "{{ item.router_id }}", + "sourceAddress": "{{ item.sourceaddress | default('UseNodeIP') }}" + }} + register: output + retries: 4 + until: output.rc == 0 + delay: "{{ retry_stagger | random + 3 }}" + with_items: + - "{{ peers | selectattr('scope', 'undefined') | list | default([]) | union(peers | selectattr('scope', 'defined') | selectattr('scope', 'equalto', 'node') | list | default([])) }}" + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: + - inventory_hostname in groups['k8s_cluster'] diff --git a/kubespray/roles/network_plugin/calico/tasks/pre.yml b/kubespray/roles/network_plugin/calico/tasks/pre.yml new file mode 100644 index 0000000..969699f --- /dev/null +++ b/kubespray/roles/network_plugin/calico/tasks/pre.yml @@ -0,0 +1,47 @@ +--- +- name: Slurp CNI config + slurp: + src: /etc/cni/net.d/10-calico.conflist + register: calico_cni_config_slurp + failed_when: false + +- name: Gather calico facts + tags: + - facts + when: calico_cni_config_slurp.content is defined + block: + - name: Set fact calico_cni_config from slurped CNI config + set_fact: + calico_cni_config: "{{ calico_cni_config_slurp['content'] | b64decode | from_json }}" + - name: Set fact calico_datastore to etcd if needed + set_fact: + calico_datastore: etcd + when: + - "'plugins' in calico_cni_config" + - "'etcd_endpoints' in calico_cni_config.plugins.0" + +- name: Calico | Get kubelet hostname + shell: >- + set -o pipefail && {{ kubectl }} get node -o custom-columns='NAME:.metadata.name,INTERNAL-IP:.status.addresses[?(@.type=="InternalIP")].address' + | egrep "{{ ansible_all_ipv4_addresses | join('$|') }}$" | cut -d" " -f1 + args: + executable: /bin/bash + register: calico_kubelet_name + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: + - "cloud_provider is defined" + +- name: Calico | Gather os specific variables + include_vars: "{{ item }}" + with_first_found: + - files: + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_release }}.yml" + - "{{ ansible_distribution | lower }}-{{ ansible_distribution_major_version | lower | replace('/', '_') }}.yml" + - "{{ ansible_distribution | lower }}.yml" + - "{{ ansible_os_family | lower }}-{{ ansible_architecture }}.yml" + - "{{ ansible_os_family | lower }}.yml" + - defaults.yml + paths: + - ../vars + skip: true diff --git a/kubespray/roles/network_plugin/calico/tasks/repos.yml b/kubespray/roles/network_plugin/calico/tasks/repos.yml new file mode 100644 index 0000000..dd29f45 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/tasks/repos.yml @@ -0,0 +1,21 @@ +--- +- name: Calico | Add wireguard yum repo + when: + - calico_wireguard_enabled + block: + + - name: Calico | Add wireguard yum repo + yum_repository: + name: copr:copr.fedorainfracloud.org:jdoss:wireguard + file: _copr:copr.fedorainfracloud.org:jdoss:wireguard + description: Copr repo for wireguard owned by jdoss + baseurl: "{{ calico_wireguard_repo }}" + gpgcheck: yes + gpgkey: https://download.copr.fedorainfracloud.org/results/jdoss/wireguard/pubkey.gpg + skip_if_unavailable: yes + enabled: yes + repo_gpgcheck: no + when: + - ansible_os_family in ['RedHat'] + - ansible_distribution not in ['Fedora'] + - ansible_facts['distribution_major_version'] | int < 9 diff --git a/kubespray/roles/network_plugin/calico/tasks/reset.yml b/kubespray/roles/network_plugin/calico/tasks/reset.yml new file mode 100644 index 0000000..8dab214 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/tasks/reset.yml @@ -0,0 +1,30 @@ +--- +- name: Reset | check vxlan.calico network device + stat: + path: /sys/class/net/vxlan.calico + get_attributes: no + get_checksum: no + get_mime: no + register: vxlan + +- name: Reset | remove the network vxlan.calico device created by calico + command: ip link del vxlan.calico + when: vxlan.stat.exists + +- name: Reset | check dummy0 network device + stat: + path: /sys/class/net/dummy0 + get_attributes: no + get_checksum: no + get_mime: no + register: dummy0 + +- name: Reset | remove the network device created by calico + command: ip link del dummy0 + when: dummy0.stat.exists + +- name: Reset | get and remove remaining routes set by bird + shell: set -o pipefail && ip route show proto bird | xargs -i bash -c "ip route del {} proto bird " + args: + executable: /bin/bash + changed_when: false diff --git a/kubespray/roles/network_plugin/calico/tasks/typha_certs.yml b/kubespray/roles/network_plugin/calico/tasks/typha_certs.yml new file mode 100644 index 0000000..5d3f279 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/tasks/typha_certs.yml @@ -0,0 +1,51 @@ +--- +- name: Calico | Check if typha-server exists + command: "{{ kubectl }} -n kube-system get secret typha-server" + register: typha_server_secret + changed_when: false + failed_when: false + +- name: Calico | Ensure calico certs dir + file: + path: /etc/calico/certs + state: directory + mode: 0755 + when: typha_server_secret.rc != 0 + +- name: Calico | Copy ssl script for typha certs + template: + src: make-ssl-calico.sh.j2 + dest: "{{ bin_dir }}/make-ssl-typha.sh" + mode: 0755 + when: typha_server_secret.rc != 0 + +- name: Calico | Copy ssl config for typha certs + copy: + src: openssl.conf + dest: /etc/calico/certs/openssl.conf + mode: 0644 + when: typha_server_secret.rc != 0 + +- name: Calico | Generate typha certs + command: >- + {{ bin_dir }}/make-ssl-typha.sh + -f /etc/calico/certs/openssl.conf + -c {{ kube_cert_dir }} + -d /etc/calico/certs + -s typha + when: typha_server_secret.rc != 0 + +- name: Calico | Create typha tls secrets + command: >- + {{ kubectl }} -n kube-system + create secret tls {{ item.name }} + --cert {{ item.cert }} + --key {{ item.key }} + with_items: + - name: typha-server + cert: /etc/calico/certs/typha-server.crt + key: /etc/calico/certs/typha-server.key + - name: typha-client + cert: /etc/calico/certs/typha-client.crt + key: /etc/calico/certs/typha-client.key + when: typha_server_secret.rc != 0 diff --git a/kubespray/roles/network_plugin/calico/templates/calico-apiserver-ns.yml.j2 b/kubespray/roles/network_plugin/calico/templates/calico-apiserver-ns.yml.j2 new file mode 100644 index 0000000..a1bdfcb --- /dev/null +++ b/kubespray/roles/network_plugin/calico/templates/calico-apiserver-ns.yml.j2 @@ -0,0 +1,10 @@ +# This is a tech-preview manifest which installs the Calico API server. Note that this manifest is liable to change +# or be removed in future releases without further warning. +# +# Namespace and namespace-scoped resources. +apiVersion: v1 +kind: Namespace +metadata: + labels: + name: calico-apiserver + name: calico-apiserver diff --git a/kubespray/roles/network_plugin/calico/templates/calico-apiserver.yml.j2 b/kubespray/roles/network_plugin/calico/templates/calico-apiserver.yml.j2 new file mode 100644 index 0000000..2429344 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/templates/calico-apiserver.yml.j2 @@ -0,0 +1,290 @@ +# Policy to ensure the API server isn't cut off. Can be modified, but ensure +# that the main API server is always able to reach the Calico API server. +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: allow-apiserver + namespace: calico-apiserver +spec: + podSelector: + matchLabels: + apiserver: "true" + ingress: + - ports: + - protocol: TCP + port: 5443 + +--- + +apiVersion: v1 +kind: Service +metadata: + name: calico-api + namespace: calico-apiserver +spec: + ports: + - name: apiserver + port: 443 + protocol: TCP + targetPort: 5443 + selector: + apiserver: "true" + type: ClusterIP + +--- + +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + apiserver: "true" + k8s-app: calico-apiserver + name: calico-apiserver + namespace: calico-apiserver +spec: + replicas: 1 + selector: + matchLabels: + apiserver: "true" + strategy: + type: Recreate + template: + metadata: + labels: + apiserver: "true" + k8s-app: calico-apiserver + name: calico-apiserver + namespace: calico-apiserver + spec: + containers: + - args: + - --secure-port=5443 + env: + - name: DATASTORE_TYPE + value: kubernetes + image: {{ calico_apiserver_image_repo }}:{{ calico_apiserver_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + livenessProbe: + httpGet: + path: /version + port: 5443 + scheme: HTTPS + initialDelaySeconds: 90 + periodSeconds: 10 + name: calico-apiserver + readinessProbe: + exec: + command: + - /code/filecheck + failureThreshold: 5 + initialDelaySeconds: 5 + periodSeconds: 10 + securityContext: + privileged: false + runAsUser: 0 + volumeMounts: + - mountPath: /code/apiserver.local.config/certificates + name: calico-apiserver-certs + dnsPolicy: ClusterFirst + nodeSelector: + kubernetes.io/os: linux + restartPolicy: Always + serviceAccount: calico-apiserver + serviceAccountName: calico-apiserver + tolerations: + - effect: NoSchedule + key: node-role.kubernetes.io/master + - effect: NoSchedule + key: node-role.kubernetes.io/control-plane + volumes: + - name: calico-apiserver-certs + secret: + secretName: calico-apiserver-certs + +--- + +apiVersion: v1 +kind: ServiceAccount +metadata: + name: calico-apiserver + namespace: calico-apiserver + +--- + +# Cluster-scoped resources below here. +apiVersion: apiregistration.k8s.io/v1 +kind: APIService +metadata: + name: v3.projectcalico.org +spec: + group: projectcalico.org + groupPriorityMinimum: 1500 + caBundle: {{ calico_apiserver_cabundle }} + service: + name: calico-api + namespace: calico-apiserver + port: 443 + version: v3 + versionPriority: 200 + +--- + +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: calico-crds +rules: +- apiGroups: + - extensions + - networking.k8s.io + - "" + resources: + - networkpolicies + - nodes + - namespaces + - pods + - serviceaccounts + verbs: + - get + - list + - watch +- apiGroups: + - crd.projectcalico.org + resources: + - globalnetworkpolicies + - networkpolicies + - clusterinformations + - hostendpoints + - globalnetworksets + - networksets + - bgpconfigurations + - bgppeers + - felixconfigurations + - kubecontrollersconfigurations + - ippools + - ipamconfigs + - ipreservations + - ipamblocks + - blockaffinities + - caliconodestatuses + verbs: + - get + - list + - watch + - create + - update + - delete +- apiGroups: + - policy + resourceNames: + - calico-apiserver + resources: + - podsecuritypolicies + verbs: + - use + +--- + +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: calico-extension-apiserver-auth-access +rules: +- apiGroups: + - "" + resourceNames: + - extension-apiserver-authentication + resources: + - configmaps + verbs: + - list + - watch + - get +- apiGroups: + - rbac.authorization.k8s.io + resources: + - clusterroles + - clusterrolebindings + - roles + - rolebindings + verbs: + - get + - list + - watch + +--- + +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: calico-webhook-reader +rules: +- apiGroups: + - admissionregistration.k8s.io + resources: + - mutatingwebhookconfigurations + - validatingwebhookconfigurations + verbs: + - get + - list + - watch + +--- + +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: calico-apiserver-access-crds +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: calico-crds +subjects: +- kind: ServiceAccount + name: calico-apiserver + namespace: calico-apiserver + +--- + +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: calico-apiserver-delegate-auth +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: system:auth-delegator +subjects: +- kind: ServiceAccount + name: calico-apiserver + namespace: calico-apiserver + +--- + +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: calico-apiserver-webhook-reader +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: calico-webhook-reader +subjects: +- kind: ServiceAccount + name: calico-apiserver + namespace: calico-apiserver + +--- + +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: calico-extension-apiserver-auth-access +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: calico-extension-apiserver-auth-access +subjects: +- kind: ServiceAccount + name: calico-apiserver + namespace: calico-apiserver diff --git a/kubespray/roles/network_plugin/calico/templates/calico-config.yml.j2 b/kubespray/roles/network_plugin/calico/templates/calico-config.yml.j2 new file mode 100644 index 0000000..4012ef7 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/templates/calico-config.yml.j2 @@ -0,0 +1,111 @@ +kind: ConfigMap +apiVersion: v1 +metadata: + name: calico-config + namespace: kube-system +data: +{% if calico_datastore == "etcd" %} + etcd_endpoints: "{{ etcd_access_addresses }}" + etcd_ca: "/calico-secrets/ca_cert.crt" + etcd_cert: "/calico-secrets/cert.crt" + etcd_key: "/calico-secrets/key.pem" +{% elif calico_datastore == "kdd" and typha_enabled %} + # To enable Typha, set this to "calico-typha" *and* set a non-zero value for Typha replicas + # below. We recommend using Typha if you have more than 50 nodes. Above 100 nodes it is + # essential. + typha_service_name: "calico-typha" +{% endif %} +{% if calico_network_backend == 'bird' %} + cluster_type: "kubespray,bgp" + calico_backend: "bird" +{% else %} + cluster_type: "kubespray" + calico_backend: "{{ calico_network_backend }}" +{% endif %} +{% if inventory_hostname in groups['k8s_cluster'] and peer_with_router | default(false) %} + as: "{{ local_as | default(global_as_num) }}" +{% endif -%} + # The CNI network configuration to install on each node. The special + # values in this config will be automatically populated. + cni_network_config: |- + { + "name": "{{ calico_cni_name }}", + "cniVersion":"0.3.1", + "plugins":[ + { + {% if calico_datastore == "kdd" %} + "datastore_type": "kubernetes", + "nodename": "__KUBERNETES_NODE_NAME__", + {% else %} + {% if cloud_provider is defined %} + "nodename": "{{ calico_kubelet_name.stdout }}", + {% else %} + "nodename": "{{ calico_baremetal_nodename }}", + {% endif %} + {% endif %} + "type": "calico", + "log_level": "info", + {% if calico_cni_log_file_path %} + "log_file_path": "{{ calico_cni_log_file_path }}", + {% endif %} + {% if calico_datastore == "etcd" %} + "etcd_endpoints": "{{ etcd_access_addresses }}", + "etcd_cert_file": "{{ calico_cert_dir }}/cert.crt", + "etcd_key_file": "{{ calico_cert_dir }}/key.pem", + "etcd_ca_cert_file": "{{ calico_cert_dir }}/ca_cert.crt", + {% endif %} + {% if calico_ipam_host_local is defined %} + "ipam": { + "type": "host-local", + "subnet": "usePodCidr" + }, + {% else %} + "ipam": { + "type": "calico-ipam", + {% if enable_dual_stack_networks %} + "assign_ipv6": "true", + {% endif %} + "assign_ipv4": "true" + }, + {% endif %} + {% if calico_allow_ip_forwarding %} + "container_settings": { + "allow_ip_forwarding": true + }, + {% endif %} + {% if (calico_feature_control is defined) and (calico_feature_control | length > 0) %} + "feature_control": { + {% for fc in calico_feature_control -%} + {% set fcval = calico_feature_control[fc] -%} + "{{ fc }}": {{ (fcval | string | lower) if (fcval == true or fcval == false) else "\"" + fcval + "\"" }}{{ "," if not loop.last else "" }} + {% endfor -%} + {{- "" }} + }, + {% endif %} + {% if enable_network_policy %} + "policy": { + "type": "k8s" + }, + {% endif %} + {% if calico_mtu is defined and calico_mtu is number %} + "mtu": {{ calico_mtu }}, + {% endif %} + "kubernetes": { + "kubeconfig": "__KUBECONFIG_FILEPATH__" + } + }, + { + "type":"portmap", + "capabilities": { + "portMappings": true + } + }, + { + "type":"bandwidth", + "capabilities": { + "bandwidth": true + } + } + ] + } + diff --git a/kubespray/roles/network_plugin/calico/templates/calico-cr.yml.j2 b/kubespray/roles/network_plugin/calico/templates/calico-cr.yml.j2 new file mode 100644 index 0000000..d00c9e9 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/templates/calico-cr.yml.j2 @@ -0,0 +1,203 @@ +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: calico-cni-plugin +rules: + - apiGroups: [""] + resources: + - pods + - nodes + - namespaces + verbs: + - get + - apiGroups: [""] + resources: + - pods/status + verbs: + - patch + - apiGroups: ["crd.projectcalico.org"] + resources: + - blockaffinities + - ipamblocks + - ipamhandles + - clusterinformations + - ippools + - ipreservations + - ipamconfigs + verbs: + - get + - list + - create + - update + - delete +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: calico-node + namespace: kube-system +rules: + - apiGroups: [""] + resources: + - pods + - nodes + - namespaces + - configmaps + verbs: + - get + # EndpointSlices are used for Service-based network policy rule + # enforcement. + - apiGroups: ["discovery.k8s.io"] + resources: + - endpointslices + verbs: + - watch + - list + - apiGroups: [""] + resources: + - endpoints + - services + verbs: + - watch + - list +{% if calico_datastore == "kdd" %} + # Used to discover Typhas. + - get +{% endif %} + - apiGroups: [""] + resources: + - nodes/status + verbs: + # Needed for clearing NodeNetworkUnavailable flag. + - patch +{% if calico_datastore == "etcd" %} + - apiGroups: + - policy + resourceNames: + - privileged + resources: + - podsecuritypolicies + verbs: + - use +{% elif calico_datastore == "kdd" %} + # Calico stores some configuration information in node annotations. + - update + # Watch for changes to Kubernetes NetworkPolicies. + - apiGroups: ["networking.k8s.io"] + resources: + - networkpolicies + verbs: + - watch + - list + # Used by Calico for policy information. + - apiGroups: [""] + resources: + - pods + - namespaces + - serviceaccounts + verbs: + - list + - watch + # The CNI plugin patches pods/status. + - apiGroups: [""] + resources: + - pods/status + verbs: + - patch + # Calico monitors various CRDs for config. + - apiGroups: ["crd.projectcalico.org"] + resources: + - globalfelixconfigs + - felixconfigurations + - bgppeers + - bgpfilters + - globalbgpconfigs + - bgpconfigurations + - ippools + - ipreservations + - ipamblocks + - globalnetworkpolicies + - globalnetworksets + - networkpolicies + - networksets + - clusterinformations + - hostendpoints + - blockaffinities + - caliconodestatuses + verbs: + - get + - list + - watch + # Calico must create and update some CRDs on startup. + - apiGroups: ["crd.projectcalico.org"] + resources: + - ippools + - felixconfigurations + - clusterinformations + verbs: + - create + - update + # Calico must update some CRDs. + - apiGroups: [ "crd.projectcalico.org" ] + resources: + - caliconodestatuses + verbs: + - update + # Calico stores some configuration information on the node. + - apiGroups: [""] + resources: + - nodes + verbs: + - get + - list + - watch + # These permissions are only required for upgrade from v2.6, and can + # be removed after upgrade or on fresh installations. + - apiGroups: ["crd.projectcalico.org"] + resources: + - bgpconfigurations + - bgppeers + verbs: + - create + - update + # These permissions are required for Calico CNI to perform IPAM allocations. + - apiGroups: ["crd.projectcalico.org"] + resources: + - blockaffinities + - ipamblocks + - ipamhandles + verbs: + - get + - list + - create + - update + - delete + - apiGroups: ["crd.projectcalico.org"] + resources: + - ipamconfigs + verbs: + - get + - create + # Block affinities must also be watchable by confd for route aggregation. + - apiGroups: ["crd.projectcalico.org"] + resources: + - blockaffinities + verbs: + - watch + # The Calico IPAM migration needs to get daemonsets. These permissions can be + # removed if not upgrading from an installation using host-local IPAM. + - apiGroups: ["apps"] + resources: + - daemonsets + verbs: + - get +{% endif %} + # Used for creating service account tokens to be used by the CNI plugin + - apiGroups: [""] + resources: + - serviceaccounts/token + resourceNames: + - calico-cni-plugin + verbs: + - create diff --git a/kubespray/roles/network_plugin/calico/templates/calico-crb.yml.j2 b/kubespray/roles/network_plugin/calico/templates/calico-crb.yml.j2 new file mode 100644 index 0000000..add99ba --- /dev/null +++ b/kubespray/roles/network_plugin/calico/templates/calico-crb.yml.j2 @@ -0,0 +1,28 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: calico-node +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: calico-node +subjects: +- kind: ServiceAccount + name: calico-node + namespace: kube-system + +--- + +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: calico-cni-plugin +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: calico-cni-plugin +subjects: +- kind: ServiceAccount + name: calico-cni-plugin + namespace: kube-system diff --git a/kubespray/roles/network_plugin/calico/templates/calico-ipamconfig.yml.j2 b/kubespray/roles/network_plugin/calico/templates/calico-ipamconfig.yml.j2 new file mode 100644 index 0000000..af7e211 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/templates/calico-ipamconfig.yml.j2 @@ -0,0 +1,8 @@ +apiVersion: crd.projectcalico.org/v1 +kind: IPAMConfig +metadata: + name: default +spec: + autoAllocateBlocks: {{ calico_ipam_autoallocateblocks }} + strictAffinity: {{ calico_ipam_strictaffinity }} + maxBlocksPerHost: {{ calico_ipam_maxblocksperhost }} diff --git a/kubespray/roles/network_plugin/calico/templates/calico-node-sa.yml.j2 b/kubespray/roles/network_plugin/calico/templates/calico-node-sa.yml.j2 new file mode 100644 index 0000000..0743303 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/templates/calico-node-sa.yml.j2 @@ -0,0 +1,13 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: calico-node + namespace: kube-system + +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: calico-cni-plugin + namespace: kube-system diff --git a/kubespray/roles/network_plugin/calico/templates/calico-node.yml.j2 b/kubespray/roles/network_plugin/calico/templates/calico-node.yml.j2 new file mode 100644 index 0000000..4e49f3b --- /dev/null +++ b/kubespray/roles/network_plugin/calico/templates/calico-node.yml.j2 @@ -0,0 +1,467 @@ +--- +# This manifest installs the calico/node container, as well +# as the Calico CNI plugins and network config on +# each master and worker node in a Kubernetes cluster. +kind: DaemonSet +apiVersion: apps/v1 +metadata: + name: calico-node + namespace: kube-system + labels: + k8s-app: calico-node +spec: + selector: + matchLabels: + k8s-app: calico-node + template: + metadata: + labels: + k8s-app: calico-node + annotations: +{% if calico_datastore == "etcd" %} + kubespray.etcd-cert/serial: "{{ etcd_client_cert_serial }}" +{% endif %} +{% if calico_felix_prometheusmetricsenabled %} + prometheus.io/scrape: 'true' + prometheus.io/port: "{{ calico_felix_prometheusmetricsport }}" +{% endif %} + spec: + nodeSelector: + {{ calico_ds_nodeselector }} + priorityClassName: system-node-critical + hostNetwork: true + serviceAccountName: calico-node + tolerations: + - operator: Exists + # Minimize downtime during a rolling upgrade or deletion; tell Kubernetes to do a "force + # deletion": https://kubernetes.io/docs/concepts/workloads/pods/pod/#termination-of-pods. + terminationGracePeriodSeconds: 0 + initContainers: +{% if calico_datastore == "kdd" %} + # This container performs upgrade from host-local IPAM to calico-ipam. + # It can be deleted if this is a fresh installation, or if you have already + # upgraded to use calico-ipam. + - name: upgrade-ipam + image: {{ calico_cni_image_repo }}:{{ calico_cni_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + command: ["/opt/cni/bin/calico-ipam", "-upgrade"] + envFrom: + - configMapRef: + # Allow KUBERNETES_SERVICE_HOST and KUBERNETES_SERVICE_PORT to be overridden for eBPF mode. + name: kubernetes-services-endpoint + optional: true + env: + - name: KUBERNETES_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: CALICO_NETWORKING_BACKEND + valueFrom: + configMapKeyRef: + name: calico-config + key: calico_backend + volumeMounts: + - mountPath: /var/lib/cni/networks + name: host-local-net-dir + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + securityContext: + privileged: true +{% endif %} + # This container installs the Calico CNI binaries + # and CNI network config file on each node. + - name: install-cni + image: {{ calico_cni_image_repo }}:{{ calico_cni_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + command: ["/opt/cni/bin/install"] + envFrom: + - configMapRef: + # Allow KUBERNETES_SERVICE_HOST and KUBERNETES_SERVICE_PORT to be overridden for eBPF mode. + name: kubernetes-services-endpoint + optional: true + env: + # The CNI network config to install on each node. + - name: CNI_NETWORK_CONFIG + valueFrom: + configMapKeyRef: + name: calico-config + key: cni_network_config + # Name of the CNI config file to create. + - name: CNI_CONF_NAME + value: "10-calico.conflist" + # Install CNI binaries + - name: UPDATE_CNI_BINARIES + value: "true" + # Prevents the container from sleeping forever. + - name: SLEEP + value: "false" +{% if calico_datastore == "kdd" %} + # Set the hostname based on the k8s node name. + - name: KUBERNETES_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName +{% endif %} + volumeMounts: + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + securityContext: + privileged: true + # Adds a Flex Volume Driver that creates a per-pod Unix Domain Socket to allow Dikastes + # to communicate with Felix over the Policy Sync API. + - name: flexvol-driver + image: {{ calico_flexvol_image_repo }}:{{ calico_flexvol_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + volumeMounts: + - name: flexvol-driver-host + mountPath: /host/driver + securityContext: + privileged: true + containers: + # Runs calico/node container on each Kubernetes node. This + # container programs network policy and routes on each + # host. + - name: calico-node + image: {{ calico_node_image_repo }}:{{ calico_node_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + envFrom: + - configMapRef: + # Allow KUBERNETES_SERVICE_HOST and KUBERNETES_SERVICE_PORT to be overridden for eBPF mode. + name: kubernetes-services-endpoint + optional: true + env: + # The location of the Calico etcd cluster. +{% if calico_datastore == "etcd" %} + - name: ETCD_ENDPOINTS + valueFrom: + configMapKeyRef: + name: calico-config + key: etcd_endpoints + # Location of the CA certificate for etcd. + - name: ETCD_CA_CERT_FILE + valueFrom: + configMapKeyRef: + name: calico-config + key: etcd_ca + # Location of the client key for etcd. + - name: ETCD_KEY_FILE + valueFrom: + configMapKeyRef: + name: calico-config + key: etcd_key + # Location of the client certificate for etcd. + - name: ETCD_CERT_FILE + valueFrom: + configMapKeyRef: + name: calico-config + key: etcd_cert +{% elif calico_datastore == "kdd" %} + # Use Kubernetes API as the backing datastore. + - name: DATASTORE_TYPE + value: "kubernetes" +{% if typha_enabled %} + # Typha support: controlled by the ConfigMap. + - name: FELIX_TYPHAK8SSERVICENAME + valueFrom: + configMapKeyRef: + name: calico-config + key: typha_service_name +{% if typha_secure %} + - name: FELIX_TYPHACN + value: typha-server + - name: FELIX_TYPHACAFILE + value: /etc/typha-ca/ca.crt + - name: FELIX_TYPHACERTFILE + value: /etc/typha-client/typha-client.crt + - name: FELIX_TYPHAKEYFILE + value: /etc/typha-client/typha-client.key +{% endif %} +{% endif %} + # Wait for the datastore. + - name: WAIT_FOR_DATASTORE + value: "true" +{% endif %} +{% if calico_network_backend == 'vxlan' %} + - name: FELIX_VXLANVNI + value: "{{ calico_vxlan_vni }}" + - name: FELIX_VXLANPORT + value: "{{ calico_vxlan_port }}" +{% endif %} + # Choose the backend to use. + - name: CALICO_NETWORKING_BACKEND + valueFrom: + configMapKeyRef: + name: calico-config + key: calico_backend + # Cluster type to identify the deployment type + - name: CLUSTER_TYPE + valueFrom: + configMapKeyRef: + name: calico-config + key: cluster_type + # Set noderef for node controller. + - name: CALICO_K8S_NODE_REF + valueFrom: + fieldRef: + fieldPath: spec.nodeName + # Disable file logging so `kubectl logs` works. + - name: CALICO_DISABLE_FILE_LOGGING + value: "true" + # Set Felix endpoint to host default action to ACCEPT. + - name: FELIX_DEFAULTENDPOINTTOHOSTACTION + value: "{{ calico_endpoint_to_host_action | default('RETURN') }}" + - name: FELIX_HEALTHHOST + value: "{{ calico_healthhost }}" +{% if kube_proxy_mode == 'ipvs' and kube_apiserver_node_port_range is defined %} + - name: FELIX_KUBENODEPORTRANGES + value: "{{ kube_apiserver_node_port_range.split('-')[0] }}:{{ kube_apiserver_node_port_range.split('-')[1] }}" +{% endif %} + - name: FELIX_IPTABLESBACKEND + value: "{{ calico_iptables_backend }}" + - name: FELIX_IPTABLESLOCKTIMEOUTSECS + value: "{{ calico_iptables_lock_timeout_secs }}" +# should be set in etcd before deployment +# # Configure the IP Pool from which Pod IPs will be chosen. +# - name: CALICO_IPV4POOL_CIDR +# value: "{{ calico_pool_cidr | default(kube_pods_subnet) }}" + - name: CALICO_IPV4POOL_IPIP + value: "{{ calico_ipv4pool_ipip }}" + - name: FELIX_IPV6SUPPORT + value: "{{ enable_dual_stack_networks | default(false) }}" + # Set Felix logging to "info" + - name: FELIX_LOGSEVERITYSCREEN + value: "{{ calico_loglevel }}" + # Set Calico startup logging to "error" + - name: CALICO_STARTUP_LOGLEVEL + value: "{{ calico_node_startup_loglevel }}" + # Enable or disable usage report + - name: FELIX_USAGEREPORTINGENABLED + value: "{{ calico_usage_reporting }}" + # Set MTU for tunnel device used if ipip is enabled +{% if calico_mtu is defined %} + # Set MTU for tunnel device used if ipip is enabled + - name: FELIX_IPINIPMTU + value: "{{ calico_veth_mtu | default(calico_mtu) }}" + # Set MTU for the VXLAN tunnel device. + - name: FELIX_VXLANMTU + value: "{{ calico_veth_mtu | default(calico_mtu) }}" + # Set MTU for the Wireguard tunnel device. + - name: FELIX_WIREGUARDMTU + value: "{{ calico_veth_mtu | default(calico_mtu) }}" +{% endif %} + - name: FELIX_CHAININSERTMODE + value: "{{ calico_felix_chaininsertmode }}" + - name: FELIX_PROMETHEUSMETRICSENABLED + value: "{{ calico_felix_prometheusmetricsenabled }}" + - name: FELIX_PROMETHEUSMETRICSPORT + value: "{{ calico_felix_prometheusmetricsport }}" + - name: FELIX_PROMETHEUSGOMETRICSENABLED + value: "{{ calico_felix_prometheusgometricsenabled }}" + - name: FELIX_PROMETHEUSPROCESSMETRICSENABLED + value: "{{ calico_felix_prometheusprocessmetricsenabled }}" +{% if calico_ip_auto_method is defined %} + - name: IP_AUTODETECTION_METHOD + value: "{{ calico_ip_auto_method }}" +{% else %} + - name: NODEIP + valueFrom: + fieldRef: + fieldPath: status.hostIP + - name: IP_AUTODETECTION_METHOD + value: "can-reach=$(NODEIP)" +{% endif %} + - name: IP + value: "autodetect" +{% if calico_ip6_auto_method is defined and enable_dual_stack_networks %} + - name: IP6_AUTODETECTION_METHOD + value: "{{ calico_ip6_auto_method }}" +{% endif %} +{% if calico_felix_mtu_iface_pattern is defined %} + - name: FELIX_MTUIFACEPATTERN + value: "{{ calico_felix_mtu_iface_pattern }}" +{% endif %} +{% if enable_dual_stack_networks %} + - name: IP6 + value: autodetect +{% endif %} +{% if calico_use_default_route_src_ipaddr | default(false) %} + - name: FELIX_DEVICEROUTESOURCEADDRESS + valueFrom: + fieldRef: + fieldPath: status.hostIP +{% endif %} + - name: NODENAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: FELIX_HEALTHENABLED + value: "true" + - name: FELIX_IGNORELOOSERPF + value: "{{ calico_node_ignorelooserpf }}" + - name: CALICO_MANAGE_CNI + value: "true" +{% if calico_node_extra_envs is defined %} +{% for key in calico_node_extra_envs %} + - name: {{ key }} + value: "{{ calico_node_extra_envs[key] }}" +{% endfor %} +{% endif %} + securityContext: + privileged: true + resources: + limits: + cpu: {{ calico_node_cpu_limit }} + memory: {{ calico_node_memory_limit }} + requests: + cpu: {{ calico_node_cpu_requests }} + memory: {{ calico_node_memory_requests }} + lifecycle: + preStop: + exec: + command: + - /bin/calico-node + - -shutdown + livenessProbe: + exec: + command: + - /bin/calico-node + - -felix-live +{% if calico_network_backend == "bird" %} + - -bird-live +{% endif %} + periodSeconds: 10 + initialDelaySeconds: 10 + timeoutSeconds: {{ calico_node_livenessprobe_timeout | default(10) }} + failureThreshold: 6 + readinessProbe: + exec: + command: + - /bin/calico-node +{% if calico_network_backend == "bird" %} + - -bird-ready +{% endif %} + - -felix-ready + periodSeconds: 10 + timeoutSeconds: {{ calico_node_readinessprobe_timeout | default(10) }} + failureThreshold: 6 + volumeMounts: + - mountPath: /lib/modules + name: lib-modules + readOnly: true + - mountPath: /var/run/calico + name: var-run-calico + readOnly: false + - mountPath: /var/lib/calico + name: var-lib-calico + readOnly: false +{% if calico_datastore == "etcd" %} + - mountPath: /calico-secrets + name: etcd-certs + readOnly: true +{% endif %} + - name: xtables-lock + mountPath: /run/xtables.lock + readOnly: false + # For maintaining CNI plugin API credentials. + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + readOnly: false +{% if typha_secure %} + - name: typha-client + mountPath: /etc/typha-client + readOnly: true + - name: typha-cacert + subPath: ca.crt + mountPath: /etc/typha-ca/ca.crt + readOnly: true +{% endif %} + - name: policysync + mountPath: /var/run/nodeagent +{% if calico_bpf_enabled %} + # For eBPF mode, we need to be able to mount the BPF filesystem at /sys/fs/bpf so we mount in the + # parent directory. + - name: sysfs + mountPath: /sys/fs/ + # Bidirectional means that, if we mount the BPF filesystem at /sys/fs/bpf it will propagate to the host. + # If the host is known to mount that filesystem already then Bidirectional can be omitted. + mountPropagation: Bidirectional +{% endif %} + - name: cni-log-dir + mountPath: /var/log/calico/cni + readOnly: true + volumes: + # Used by calico/node. + - name: lib-modules + hostPath: + path: /lib/modules + - name: var-run-calico + hostPath: + path: /var/run/calico + - name: var-lib-calico + hostPath: + path: /var/lib/calico + # Used to install CNI. + - name: cni-net-dir + hostPath: + path: /etc/cni/net.d + - name: cni-bin-dir + hostPath: + path: /opt/cni/bin +{% if calico_datastore == "etcd" %} + # Mount in the etcd TLS secrets. + - name: etcd-certs + hostPath: + path: "{{ calico_cert_dir }}" +{% endif %} + # Mount the global iptables lock file, used by calico/node + - name: xtables-lock + hostPath: + path: /run/xtables.lock + type: FileOrCreate +{% if calico_datastore == "kdd" %} + # Mount in the directory for host-local IPAM allocations. This is + # used when upgrading from host-local to calico-ipam, and can be removed + # if not using the upgrade-ipam init container. + - name: host-local-net-dir + hostPath: + path: /var/lib/cni/networks +{% endif %} +{% if typha_enabled and typha_secure %} + - name: typha-client + secret: + secretName: typha-client + items: + - key: tls.crt + path: typha-client.crt + - key: tls.key + path: typha-client.key + - name: typha-cacert + hostPath: + path: "/etc/kubernetes/ssl/" +{% endif %} +{% if calico_bpf_enabled %} + - name: sysfs + hostPath: + path: /sys/fs/ + type: DirectoryOrCreate +{% endif %} + # Used to access CNI logs. + - name: cni-log-dir + hostPath: + path: /var/log/calico/cni + # Used to create per-pod Unix Domain Sockets + - name: policysync + hostPath: + type: DirectoryOrCreate + path: /var/run/nodeagent + # Used to install Flex Volume Driver + - name: flexvol-driver-host + hostPath: + type: DirectoryOrCreate + path: "{{ kubelet_flexvolumes_plugins_dir | default('/usr/libexec/kubernetes/kubelet-plugins/volume/exec') }}/nodeagent~uds" + updateStrategy: + rollingUpdate: + maxUnavailable: {{ serial | default('20%') }} + type: RollingUpdate diff --git a/kubespray/roles/network_plugin/calico/templates/calico-typha.yml.j2 b/kubespray/roles/network_plugin/calico/templates/calico-typha.yml.j2 new file mode 100644 index 0000000..22d2f2c --- /dev/null +++ b/kubespray/roles/network_plugin/calico/templates/calico-typha.yml.j2 @@ -0,0 +1,190 @@ +# This manifest creates a Service, which will be backed by Calico's Typha daemon. +# Typha sits in between Felix and the API server, reducing Calico's load on the API server. + +apiVersion: v1 +kind: Service +metadata: + name: calico-typha + namespace: kube-system + labels: + k8s-app: calico-typha +spec: + ports: + - port: 5473 + protocol: TCP + targetPort: calico-typha + name: calico-typha +{% if typha_prometheusmetricsenabled %} + - port: {{ typha_prometheusmetricsport }} + protocol: TCP + targetPort: http-metrics + name: metrics +{% endif %} + selector: + k8s-app: calico-typha + +--- + +# This manifest creates a Deployment of Typha to back the above service. + +apiVersion: apps/v1 +kind: Deployment +metadata: + name: calico-typha + namespace: kube-system + labels: + k8s-app: calico-typha +spec: + # Number of Typha replicas. To enable Typha, set this to a non-zero value *and* set the + # typha_service_name variable in the calico-config ConfigMap above. + # + # We recommend using Typha if you have more than 50 nodes. Above 100 nodes it is essential + # (when using the Kubernetes datastore). Use one replica for every 100-200 nodes. In + # production, we recommend running at least 3 replicas to reduce the impact of rolling upgrade. + replicas: {{ typha_replicas }} + revisionHistoryLimit: 2 + selector: + matchLabels: + k8s-app: calico-typha + template: + metadata: + labels: + k8s-app: calico-typha + annotations: + cluster-autoscaler.kubernetes.io/safe-to-evict: 'true' +{% if typha_prometheusmetricsenabled %} + prometheus.io/scrape: 'true' + prometheus.io/port: "{{ typha_prometheusmetricsport }}" +{% endif %} + spec: + nodeSelector: + kubernetes.io/os: linux + hostNetwork: true + tolerations: + - key: node-role.kubernetes.io/master + operator: Exists + effect: NoSchedule + - key: node-role.kubernetes.io/control-plane + operator: Exists + effect: NoSchedule + # Since Calico can't network a pod until Typha is up, we need to run Typha itself + # as a host-networked pod. + serviceAccountName: calico-node + priorityClassName: system-cluster-critical + # fsGroup allows using projected serviceaccount tokens as described here kubernetes/kubernetes#82573 + securityContext: + fsGroup: 65534 + containers: + - image: {{ calico_typha_image_repo }}:{{ calico_typha_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + name: calico-typha + ports: + - containerPort: 5473 + name: calico-typha + protocol: TCP +{% if typha_prometheusmetricsenabled %} + - containerPort: {{ typha_prometheusmetricsport }} + name: http-metrics + protocol: TCP +{% endif %} + envFrom: + - configMapRef: + # Allow KUBERNETES_SERVICE_HOST and KUBERNETES_SERVICE_PORT to be overridden for eBPF mode. + name: kubernetes-services-endpoint + optional: true + env: + # Enable "info" logging by default. Can be set to "debug" to increase verbosity. + - name: TYPHA_LOGSEVERITYSCREEN + value: "info" + # Disable logging to file and syslog since those don't make sense in Kubernetes. + - name: TYPHA_LOGFILEPATH + value: "none" + - name: TYPHA_LOGSEVERITYSYS + value: "none" + # Monitor the Kubernetes API to find the number of running instances and rebalance + # connections. + - name: TYPHA_CONNECTIONREBALANCINGMODE + value: "kubernetes" + - name: TYPHA_DATASTORETYPE + value: "kubernetes" + - name: TYPHA_HEALTHENABLED + value: "true" + - name: TYPHA_MAXCONNECTIONSLOWERLIMIT + value: "{{ typha_max_connections_lower_limit }}" +{% if typha_secure %} + - name: TYPHA_CAFILE + value: /etc/ca/ca.crt + - name: TYPHA_CLIENTCN + value: typha-client + - name: TYPHA_SERVERCERTFILE + value: /etc/typha/server_certificate.pem + - name: TYPHA_SERVERKEYFILE + value: /etc/typha/server_key.pem +{% endif %} +{% if typha_prometheusmetricsenabled %} + # Since Typha is host-networked, + # this opens a port on the host, which may need to be secured. + - name: TYPHA_PROMETHEUSMETRICSENABLED + value: "true" + - name: TYPHA_PROMETHEUSMETRICSPORT + value: "{{ typha_prometheusmetricsport }}" +{% endif %} +{% if typha_secure %} + volumeMounts: + - mountPath: /etc/typha + name: typha-server + readOnly: true + - mountPath: /etc/ca/ca.crt + subPath: ca.crt + name: cacert + readOnly: true +{% endif %} + # Needed for version >=3.7 when the 'host-local' ipam is used + # Should never happen given templates/cni-calico.conflist.j2 + # Configure route aggregation based on pod CIDR. + # - name: USE_POD_CIDR + # value: "true" + livenessProbe: + httpGet: + path: /liveness + port: 9098 + host: localhost + periodSeconds: 30 + initialDelaySeconds: 30 + readinessProbe: + httpGet: + path: /readiness + port: 9098 + host: localhost + periodSeconds: 10 +{% if typha_secure %} + volumes: + - name: typha-server + secret: + secretName: typha-server + items: + - key: tls.crt + path: server_certificate.pem + - key: tls.key + path: server_key.pem + - name: cacert + hostPath: + path: "{{ kube_cert_dir }}" +{% endif %} + +--- + +# This manifest creates a Pod Disruption Budget for Typha to allow K8s Cluster Autoscaler to evict + +apiVersion: policy/v1 +kind: PodDisruptionBudget +metadata: + name: calico-typha + namespace: kube-system + labels: + k8s-app: calico-typha +spec: + maxUnavailable: 1 + selector: + matchLabels: + k8s-app: calico-typha diff --git a/kubespray/roles/network_plugin/calico/templates/calicoctl.etcd.sh.j2 b/kubespray/roles/network_plugin/calico/templates/calicoctl.etcd.sh.j2 new file mode 100644 index 0000000..fcde4a5 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/templates/calicoctl.etcd.sh.j2 @@ -0,0 +1,6 @@ +#!/bin/bash +ETCD_ENDPOINTS={{ etcd_access_addresses }} \ +ETCD_CA_CERT_FILE={{ calico_cert_dir }}/ca_cert.crt \ +ETCD_CERT_FILE={{ calico_cert_dir }}/cert.crt \ +ETCD_KEY_FILE={{ calico_cert_dir }}/key.pem \ +{{ bin_dir }}/calicoctl --allow-version-mismatch "$@" diff --git a/kubespray/roles/network_plugin/calico/templates/calicoctl.kdd.sh.j2 b/kubespray/roles/network_plugin/calico/templates/calicoctl.kdd.sh.j2 new file mode 100644 index 0000000..ef89f39 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/templates/calicoctl.kdd.sh.j2 @@ -0,0 +1,8 @@ +#!/bin/bash +DATASTORE_TYPE=kubernetes \ +{% if inventory_hostname in groups['kube_control_plane'] %} +KUBECONFIG=/etc/kubernetes/admin.conf \ +{% else %} +KUBECONFIG=/etc/cni/net.d/calico-kubeconfig \ +{% endif %} +{{ bin_dir }}/calicoctl --allow-version-mismatch "$@" diff --git a/kubespray/roles/network_plugin/calico/templates/kubernetes-services-endpoint.yml.j2 b/kubespray/roles/network_plugin/calico/templates/kubernetes-services-endpoint.yml.j2 new file mode 100644 index 0000000..f1e8177 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/templates/kubernetes-services-endpoint.yml.j2 @@ -0,0 +1,11 @@ +--- +apiVersion: v1 +kind: ConfigMap +metadata: + namespace: kube-system + name: kubernetes-services-endpoint +data: +{% if calico_bpf_enabled %} + KUBERNETES_SERVICE_HOST: "{{ kube_apiserver_global_endpoint | urlsplit('hostname') }}" + KUBERNETES_SERVICE_PORT: "{{ kube_apiserver_global_endpoint | urlsplit('port') }}" +{% endif %} diff --git a/kubespray/roles/network_plugin/calico/templates/make-ssl-calico.sh.j2 b/kubespray/roles/network_plugin/calico/templates/make-ssl-calico.sh.j2 new file mode 100644 index 0000000..94b2022 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/templates/make-ssl-calico.sh.j2 @@ -0,0 +1,102 @@ +#!/bin/bash + +# Author: Smana smainklh@gmail.com +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +set -o errexit +set -o pipefail +usage() +{ + cat << EOF +Create self signed certificates + +Usage : $(basename $0) -f [-d ] + -h | --help : Show this message + -f | --config : Openssl configuration file + -d | --ssldir : Directory where the certificates will be installed + -c | --cadir : Directory where the existing CA is located + -s | --service : Service for the ca + + ex : + $(basename $0) -f openssl.conf -d /srv/ssl +EOF +} + +# Options parsing +while (($#)); do + case "$1" in + -h | --help) usage; exit 0;; + -f | --config) CONFIG=${2}; shift 2;; + -d | --ssldir) SSLDIR="${2}"; shift 2;; + -c | --cadir) CADIR="${2}"; shift 2;; + -s | --service) SERVICE="${2}"; shift 2;; + *) + usage + echo "ERROR : Unknown option" + exit 3 + ;; + esac +done + +if [ -z ${CONFIG} ]; then + echo "ERROR: the openssl configuration file is missing. option -f" + exit 1 +fi +if [ -z ${SSLDIR} ]; then + SSLDIR="/etc/calico/certs" +fi + +tmpdir=$(mktemp -d /tmp/calico_${SERVICE}_certs.XXXXXX) +trap 'rm -rf "${tmpdir}"' EXIT +cd "${tmpdir}" + +mkdir -p ${SSLDIR} ${CADIR} + +# Root CA +if [ -e "$CADIR/ca.key" ]; then + # Reuse existing CA + cp $CADIR/{ca.crt,ca.key} . +else + openssl genrsa -out ca.key {{certificates_key_size}} > /dev/null 2>&1 + openssl req -x509 -new -nodes -key ca.key -days {{certificates_duration}} -out ca.crt -subj "/CN=calico-${SERVICE}-ca" > /dev/null 2>&1 +fi + +if [ $SERVICE == "typha" ]; then + # Typha server + openssl genrsa -out typha-server.key {{certificates_key_size}} > /dev/null 2>&1 + openssl req -new -key typha-server.key -out typha-server.csr -subj "/CN=typha-server" -config ${CONFIG} > /dev/null 2>&1 + openssl x509 -req -in typha-server.csr -CA ca.crt -CAkey ca.key -CAcreateserial -out typha-server.crt -days {{certificates_duration}} -extensions ssl_client -extfile ${CONFIG} > /dev/null 2>&1 + + # Typha client + openssl genrsa -out typha-client.key {{certificates_key_size}} > /dev/null 2>&1 + openssl req -new -key typha-client.key -out typha-client.csr -subj "/CN=typha-client" -config ${CONFIG} > /dev/null 2>&1 + openssl x509 -req -in typha-client.csr -CA ca.crt -CAkey ca.key -CAcreateserial -out typha-client.crt -days {{certificates_duration}} -extensions ssl_client -extfile ${CONFIG} > /dev/null 2>&1 + +elif [ $SERVICE == "apiserver" ]; then + # calico-apiserver + openssl genrsa -out apiserver.key {{certificates_key_size}} > /dev/null 2>&1 + openssl req -new -key apiserver.key -out apiserver.csr -subj "/CN=calico-apiserver" -config ${CONFIG} > /dev/null 2>&1 + openssl x509 -req -in apiserver.csr -CA ca.crt -CAkey ca.key -CAcreateserial -out apiserver.crt -days {{certificates_duration}} -extensions ssl_client_apiserver -extfile ${CONFIG} > /dev/null 2>&1 +else + echo "ERROR: the openssl configuration file is missing. option -s" + exit 1 +fi + +# Install certs +if [ -e "$CADIR/ca.key" ]; then + # No pass existing CA + rm -f ca.crt ca.key +fi + +mv {*.crt,*.key} ${SSLDIR}/ diff --git a/kubespray/roles/network_plugin/calico/vars/amazon.yml b/kubespray/roles/network_plugin/calico/vars/amazon.yml new file mode 100644 index 0000000..83efdcd --- /dev/null +++ b/kubespray/roles/network_plugin/calico/vars/amazon.yml @@ -0,0 +1,5 @@ +--- +calico_wireguard_repo: https://download.copr.fedorainfracloud.org/results/jdoss/wireguard/epel-7-$basearch/ +calico_wireguard_packages: + - wireguard-dkms + - wireguard-tools diff --git a/kubespray/roles/network_plugin/calico/vars/centos-9.yml b/kubespray/roles/network_plugin/calico/vars/centos-9.yml new file mode 100644 index 0000000..43df545 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/vars/centos-9.yml @@ -0,0 +1,3 @@ +--- +calico_wireguard_packages: + - wireguard-tools diff --git a/kubespray/roles/network_plugin/calico/vars/debian.yml b/kubespray/roles/network_plugin/calico/vars/debian.yml new file mode 100644 index 0000000..baf603c --- /dev/null +++ b/kubespray/roles/network_plugin/calico/vars/debian.yml @@ -0,0 +1,3 @@ +--- +calico_wireguard_packages: + - wireguard diff --git a/kubespray/roles/network_plugin/calico/vars/fedora.yml b/kubespray/roles/network_plugin/calico/vars/fedora.yml new file mode 100644 index 0000000..43df545 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/vars/fedora.yml @@ -0,0 +1,3 @@ +--- +calico_wireguard_packages: + - wireguard-tools diff --git a/kubespray/roles/network_plugin/calico/vars/opensuse.yml b/kubespray/roles/network_plugin/calico/vars/opensuse.yml new file mode 100644 index 0000000..43df545 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/vars/opensuse.yml @@ -0,0 +1,3 @@ +--- +calico_wireguard_packages: + - wireguard-tools diff --git a/kubespray/roles/network_plugin/calico/vars/redhat-9.yml b/kubespray/roles/network_plugin/calico/vars/redhat-9.yml new file mode 100644 index 0000000..43df545 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/vars/redhat-9.yml @@ -0,0 +1,3 @@ +--- +calico_wireguard_packages: + - wireguard-tools diff --git a/kubespray/roles/network_plugin/calico/vars/redhat.yml b/kubespray/roles/network_plugin/calico/vars/redhat.yml new file mode 100644 index 0000000..a83a8a5 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/vars/redhat.yml @@ -0,0 +1,4 @@ +--- +calico_wireguard_packages: + - wireguard-dkms + - wireguard-tools diff --git a/kubespray/roles/network_plugin/calico/vars/rocky-9.yml b/kubespray/roles/network_plugin/calico/vars/rocky-9.yml new file mode 100644 index 0000000..43df545 --- /dev/null +++ b/kubespray/roles/network_plugin/calico/vars/rocky-9.yml @@ -0,0 +1,3 @@ +--- +calico_wireguard_packages: + - wireguard-tools diff --git a/kubespray/roles/network_plugin/cilium/defaults/main.yml b/kubespray/roles/network_plugin/cilium/defaults/main.yml new file mode 100644 index 0000000..f4c70e4 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/defaults/main.yml @@ -0,0 +1,311 @@ +--- +cilium_min_version_required: "1.10" +# Log-level +cilium_debug: false + +cilium_mtu: "" +cilium_enable_ipv4: true +cilium_enable_ipv6: false + +# Cilium agent health port +cilium_agent_health_port: "{%- if cilium_version | regex_replace('v') is version('1.11.6', '>=') -%}9879{%- else -%}9876{%- endif -%}" + +# Identity allocation mode selects how identities are shared between cilium +# nodes by setting how they are stored. The options are "crd" or "kvstore". +# - "crd" stores identities in kubernetes as CRDs (custom resource definition). +# These can be queried with: +# `kubectl get ciliumid` +# - "kvstore" stores identities in an etcd kvstore. +# - In order to support External Workloads, "crd" is required +# - Ref: https://docs.cilium.io/en/stable/gettingstarted/external-workloads/#setting-up-support-for-external-workloads-beta +# - KVStore operations are only required when cilium-operator is running with any of the below options: +# - --synchronize-k8s-services +# - --synchronize-k8s-nodes +# - --identity-allocation-mode=kvstore +# - Ref: https://docs.cilium.io/en/stable/internals/cilium_operator/#kvstore-operations +cilium_identity_allocation_mode: kvstore + +# Etcd SSL dirs +cilium_cert_dir: /etc/cilium/certs +kube_etcd_cacert_file: ca.pem +kube_etcd_cert_file: node-{{ inventory_hostname }}.pem +kube_etcd_key_file: node-{{ inventory_hostname }}-key.pem + +# Limits for apps +cilium_memory_limit: 500M +cilium_cpu_limit: 500m +cilium_memory_requests: 64M +cilium_cpu_requests: 100m + +# Overlay Network Mode +cilium_tunnel_mode: vxlan +# Optional features +cilium_enable_prometheus: false +# Enable if you want to make use of hostPort mappings +cilium_enable_portmap: false +# Monitor aggregation level (none/low/medium/maximum) +cilium_monitor_aggregation: medium +# Kube Proxy Replacement mode (strict/partial) +cilium_kube_proxy_replacement: partial + +# If upgrading from Cilium < 1.5, you may want to override some of these options +# to prevent service disruptions. See also: +# http://docs.cilium.io/en/stable/install/upgrade/#changes-that-may-require-action +cilium_preallocate_bpf_maps: false + +# `cilium_tofqdns_enable_poller` is deprecated in 1.8, removed in 1.9 +cilium_tofqdns_enable_poller: false + +# `cilium_enable_legacy_services` is deprecated in 1.6, removed in 1.9 +cilium_enable_legacy_services: false + +# Deploy cilium even if kube_network_plugin is not cilium. +# This enables to deploy cilium alongside another CNI to replace kube-proxy. +cilium_deploy_additionally: false + +# Auto direct nodes routes can be used to advertise pods routes in your cluster +# without any tunelling (with `cilium_tunnel_mode` sets to `disabled`). +# This works only if you have a L2 connectivity between all your nodes. +# You wil also have to specify the variable `cilium_native_routing_cidr` to +# make this work. Please refer to the cilium documentation for more +# information about this kind of setups. +cilium_auto_direct_node_routes: false + +# Allows to explicitly specify the IPv4 CIDR for native routing. +# When specified, Cilium assumes networking for this CIDR is preconfigured and +# hands traffic destined for that range to the Linux network stack without +# applying any SNAT. +# Generally speaking, specifying a native routing CIDR implies that Cilium can +# depend on the underlying networking stack to route packets to their +# destination. To offer a concrete example, if Cilium is configured to use +# direct routing and the Kubernetes CIDR is included in the native routing CIDR, +# the user must configure the routes to reach pods, either manually or by +# setting the auto-direct-node-routes flag. +cilium_native_routing_cidr: "" + +# Allows to explicitly specify the IPv6 CIDR for native routing. +cilium_native_routing_cidr_ipv6: "" + +# Enable transparent network encryption. +cilium_encryption_enabled: false + +# Encryption method. Can be either ipsec or wireguard. +# Only effective when `cilium_encryption_enabled` is set to true. +cilium_encryption_type: "ipsec" + +# Enable encryption for pure node to node traffic. +# This option is only effective when `cilium_encryption_type` is set to `ipsec`. +cilium_ipsec_node_encryption: false + +# If your kernel or distribution does not support WireGuard, Cilium agent can be configured to fall back on the user-space implementation. +# When this flag is enabled and Cilium detects that the kernel has no native support for WireGuard, +# it will fallback on the wireguard-go user-space implementation of WireGuard. +# This option is only effective when `cilium_encryption_type` is set to `wireguard`. +cilium_wireguard_userspace_fallback: false + +# Enable Bandwidth Manager +# Cilium’s bandwidth manager supports the kubernetes.io/egress-bandwidth Pod annotation. +# Bandwidth enforcement currently does not work in combination with L7 Cilium Network Policies. +# In case they select the Pod at egress, then the bandwidth enforcement will be disabled for those Pods. +# Bandwidth Manager requires a v5.1.x or more recent Linux kernel. +cilium_enable_bandwidth_manager: false + +# IP Masquerade Agent +# https://docs.cilium.io/en/stable/concepts/networking/masquerading/ +# By default, all packets from a pod destined to an IP address outside of the cilium_native_routing_cidr range are masqueraded +cilium_ip_masq_agent_enable: false + +### A packet sent from a pod to a destination which belongs to any CIDR from the nonMasqueradeCIDRs is not going to be masqueraded +cilium_non_masquerade_cidrs: + - 10.0.0.0/8 + - 172.16.0.0/12 + - 192.168.0.0/16 + - 100.64.0.0/10 + - 192.0.0.0/24 + - 192.0.2.0/24 + - 192.88.99.0/24 + - 198.18.0.0/15 + - 198.51.100.0/24 + - 203.0.113.0/24 + - 240.0.0.0/4 +### Indicates whether to masquerade traffic to the link local prefix. +### If the masqLinkLocal is not set or set to false, then 169.254.0.0/16 is appended to the non-masquerade CIDRs list. +cilium_masq_link_local: false +### A time interval at which the agent attempts to reload config from disk +cilium_ip_masq_resync_interval: 60s + +# Hubble +### Enable Hubble without install +cilium_enable_hubble: false +### Enable Hubble Metrics +cilium_enable_hubble_metrics: false +### if cilium_enable_hubble_metrics: true +cilium_hubble_metrics: {} +# - dns +# - drop +# - tcp +# - flow +# - icmp +# - http +### Enable Hubble install +cilium_hubble_install: false +### Enable auto generate certs if cilium_hubble_install: true +cilium_hubble_tls_generate: false + +# The default IP address management mode is "Cluster Scope". +# https://docs.cilium.io/en/stable/concepts/networking/ipam/ +cilium_ipam_mode: cluster-pool + +# Cluster Pod CIDRs use the kube_pods_subnet value by default. +# If your node network is in the same range you will lose connectivity to other nodes. +# Defaults to kube_pods_subnet if not set. +# cilium_pool_cidr: 10.233.64.0/18 + +# When cilium_enable_ipv6 is used, you need to set the IPV6 value. Defaults to kube_pods_subnet_ipv6 if not set. +# cilium_pool_cidr_ipv6: fd85:ee78:d8a6:8607::1:0000/112 + +# When cilium IPAM uses the "Cluster Scope" mode, it will pre-allocate a segment of IP to each node, +# schedule the Pod to this node, and then allocate IP from here. cilium_pool_mask_size Specifies +# the size allocated from cluster Pod CIDR to node.ipam.podCIDRs +# Defaults to kube_network_node_prefix if not set. +# cilium_pool_mask_size: "24" + +# cilium_pool_mask_size Specifies the size allocated to node.ipam.podCIDRs from cluster Pod IPV6 CIDR +# Defaults to kube_network_node_prefix_ipv6 if not set. +# cilium_pool_mask_size_ipv6: "120" + + +# Extra arguments for the Cilium agent +cilium_agent_custom_args: [] + +# For adding and mounting extra volumes to the cilium agent +cilium_agent_extra_volumes: [] +cilium_agent_extra_volume_mounts: [] + +cilium_agent_extra_env_vars: [] + +cilium_operator_replicas: 2 + +# The address at which the cillium operator bind health check api +cilium_operator_api_serve_addr: "127.0.0.1:9234" + +## A dictionary of extra config variables to add to cilium-config, formatted like: +## cilium_config_extra_vars: +## var1: "value1" +## var2: "value2" +cilium_config_extra_vars: {} + +# For adding and mounting extra volumes to the cilium operator +cilium_operator_extra_volumes: [] +cilium_operator_extra_volume_mounts: [] + +# Extra arguments for the Cilium Operator +cilium_operator_custom_args: [] + +# Name of the cluster. Only relevant when building a mesh of clusters. +cilium_cluster_name: default + +# Make Cilium take ownership over the `/etc/cni/net.d` directory on the node, renaming all non-Cilium CNI configurations to `*.cilium_bak`. +# This ensures no Pods can be scheduled using other CNI plugins during Cilium agent downtime. +# Available for Cilium v1.10 and up. +cilium_cni_exclusive: true + +# Configure the log file for CNI logging with retention policy of 7 days. +# Disable CNI file logging by setting this field to empty explicitly. +# Available for Cilium v1.12 and up. +cilium_cni_log_file: "/var/run/cilium/cilium-cni.log" + +# -- Configure cgroup related configuration +# -- Enable auto mount of cgroup2 filesystem. +# When `cilium_cgroup_auto_mount` is enabled, cgroup2 filesystem is mounted at +# `cilium_cgroup_host_root` path on the underlying host and inside the cilium agent pod. +# If users disable `cilium_cgroup_auto_mount`, it's expected that users have mounted +# cgroup2 filesystem at the specified `cilium_cgroup_auto_mount` volume, and then the +# volume will be mounted inside the cilium agent pod at the same path. +# Available for Cilium v1.11 and up +cilium_cgroup_auto_mount: true +# -- Configure cgroup root where cgroup2 filesystem is mounted on the host +cilium_cgroup_host_root: "/run/cilium/cgroupv2" + +# Specifies the ratio (0.0-1.0) of total system memory to use for dynamic +# sizing of the TCP CT, non-TCP CT, NAT and policy BPF maps. +cilium_bpf_map_dynamic_size_ratio: "0.0025" + +# -- Enables masquerading of IPv4 traffic leaving the node from endpoints. +# Available for Cilium v1.10 and up +cilium_enable_ipv4_masquerade: true +# -- Enables masquerading of IPv6 traffic leaving the node from endpoints. +# Available for Cilium v1.10 and up +cilium_enable_ipv6_masquerade: true + +# -- Enable native IP masquerade support in eBPF +cilium_enable_bpf_masquerade: false + +# -- Configure whether direct routing mode should route traffic via +# host stack (true) or directly and more efficiently out of BPF (false) if +# the kernel supports it. The latter has the implication that it will also +# bypass netfilter in the host namespace. +cilium_enable_host_legacy_routing: true + +# -- Enable use of the remote node identity. +# ref: https://docs.cilium.io/en/v1.7/install/upgrade/#configmap-remote-node-identity +cilium_enable_remote_node_identity: true + +# -- Enable the use of well-known identities. +cilium_enable_well_known_identities: false + +# The monitor aggregation flags determine which TCP flags which, upon the +# first observation, cause monitor notifications to be generated. +# +# Only effective when monitor aggregation is set to "medium" or higher. +cilium_monitor_aggregation_flags: "all" + +cilium_enable_bpf_clock_probe: true + +# -- Whether to enable CNP status updates. +cilium_disable_cnp_status_updates: true + +# Configure how long to wait for the Cilium DaemonSet to be ready again +cilium_rolling_restart_wait_retries_count: 30 +cilium_rolling_restart_wait_retries_delay_seconds: 10 + +# Cilium changed the default metrics exporter ports in 1.12 +cilium_agent_scrape_port: "{{ cilium_version | regex_replace('v') is version('1.12', '>=') | ternary('9962', '9090') }}" +cilium_operator_scrape_port: "{{ cilium_version | regex_replace('v') is version('1.12', '>=') | ternary('9963', '6942') }}" +cilium_hubble_scrape_port: "{{ cilium_version | regex_replace('v') is version('1.12', '>=') | ternary('9965', '9091') }}" + +# Cilium certgen args for generate certificate for hubble mTLS +cilium_certgen_args: + cilium-namespace: kube-system + ca-reuse-secret: true + ca-secret-name: hubble-ca-secret + ca-generate: true + ca-validity-duration: 94608000s + hubble-server-cert-generate: true + hubble-server-cert-common-name: '*.{{ cilium_cluster_name }}.hubble-grpc.cilium.io' + hubble-server-cert-validity-duration: 94608000s + hubble-server-cert-secret-name: hubble-server-certs + hubble-relay-client-cert-generate: true + hubble-relay-client-cert-common-name: '*.{{ cilium_cluster_name }}.hubble-grpc.cilium.io' + hubble-relay-client-cert-validity-duration: 94608000s + hubble-relay-client-cert-secret-name: hubble-relay-client-certs + hubble-relay-server-cert-generate: false + +# A list of extra rules variables to add to clusterrole for cilium operator, formatted like: +# cilium_clusterrole_rules_operator_extra_vars: +# - apiGroups: +# - '""' +# resources: +# - pods +# verbs: +# - delete +# - apiGroups: +# - '""' +# resources: +# - nodes +# verbs: +# - list +# - watch +# resourceNames: +# - toto +cilium_clusterrole_rules_operator_extra_vars: [] diff --git a/kubespray/roles/network_plugin/cilium/tasks/apply.yml b/kubespray/roles/network_plugin/cilium/tasks/apply.yml new file mode 100644 index 0000000..75868ba --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/tasks/apply.yml @@ -0,0 +1,33 @@ +--- +- name: Cilium | Start Resources + kube: + name: "{{ item.item.name }}" + namespace: "kube-system" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/{{ item.item.name }}-{{ item.item.file }}" + state: "latest" + loop: "{{ cilium_node_manifests.results }}" + when: inventory_hostname == groups['kube_control_plane'][0] and not item is skipped + +- name: Cilium | Wait for pods to run + command: "{{ kubectl }} -n kube-system get pods -l k8s-app=cilium -o jsonpath='{.items[?(@.status.containerStatuses[0].ready==false)].metadata.name}'" # noqa literal-compare + register: pods_not_ready + until: pods_not_ready.stdout.find("cilium")==-1 + retries: "{{ cilium_rolling_restart_wait_retries_count | int }}" + delay: "{{ cilium_rolling_restart_wait_retries_delay_seconds | int }}" + failed_when: false + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Cilium | Hubble install + kube: + name: "{{ item.item.name }}" + namespace: "kube-system" + kubectl: "{{ bin_dir }}/kubectl" + resource: "{{ item.item.type }}" + filename: "{{ kube_config_dir }}/addons/hubble/{{ item.item.name }}-{{ item.item.file }}" + state: "latest" + loop: "{{ cilium_hubble_manifests.results }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] and not item is skipped + - cilium_enable_hubble and cilium_hubble_install diff --git a/kubespray/roles/network_plugin/cilium/tasks/check.yml b/kubespray/roles/network_plugin/cilium/tasks/check.yml new file mode 100644 index 0000000..c65591f --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/tasks/check.yml @@ -0,0 +1,63 @@ +--- +- name: Cilium | Check Cilium encryption `cilium_ipsec_key` for ipsec + assert: + that: + - "cilium_ipsec_key is defined" + msg: "cilium_ipsec_key should be defined to enable encryption using ipsec" + when: + - cilium_encryption_enabled + - cilium_encryption_type == "ipsec" + - cilium_tunnel_mode in ['vxlan'] + +# TODO: Clean this task up when we drop backward compatibility support for `cilium_ipsec_enabled` +- name: Stop if `cilium_ipsec_enabled` is defined and `cilium_encryption_type` is not `ipsec` + assert: + that: cilium_encryption_type == 'ipsec' + msg: > + It is not possible to use `cilium_ipsec_enabled` when `cilium_encryption_type` is set to {{ cilium_encryption_type }}. + when: + - cilium_ipsec_enabled is defined + - cilium_ipsec_enabled + - kube_network_plugin == 'cilium' or cilium_deploy_additionally | default(false) | bool + +- name: Stop if kernel version is too low for Cilium Wireguard encryption + assert: + that: ansible_kernel.split('-')[0] is version('5.6.0', '>=') + when: + - kube_network_plugin == 'cilium' or cilium_deploy_additionally | default(false) | bool + - cilium_encryption_enabled + - cilium_encryption_type == "wireguard" + - not ignore_assert_errors + +- name: Stop if bad Cilium identity allocation mode + assert: + that: cilium_identity_allocation_mode in ['crd', 'kvstore'] + msg: "cilium_identity_allocation_mode must be either 'crd' or 'kvstore'" + +- name: Stop if bad Cilium Cluster ID + assert: + that: + - cilium_cluster_id <= 255 + - cilium_cluster_id >= 0 + msg: "'cilium_cluster_id' must be between 1 and 255" + when: cilium_cluster_id is defined + +- name: Stop if bad encryption type + assert: + that: cilium_encryption_type in ['ipsec', 'wireguard'] + msg: "cilium_encryption_type must be either 'ipsec' or 'wireguard'" + when: cilium_encryption_enabled + +- name: Stop if cilium_version is < v1.10.0 + assert: + that: cilium_version | regex_replace('v') is version(cilium_min_version_required, '>=') + msg: "cilium_version is too low. Minimum version {{ cilium_min_version_required }}" + +# TODO: Clean this task up when we drop backward compatibility support for `cilium_ipsec_enabled` +- name: Set `cilium_encryption_type` to "ipsec" and if `cilium_ipsec_enabled` is true + set_fact: + cilium_encryption_type: ipsec + cilium_encryption_enabled: true + when: + - cilium_ipsec_enabled is defined + - cilium_ipsec_enabled diff --git a/kubespray/roles/network_plugin/cilium/tasks/install.yml b/kubespray/roles/network_plugin/cilium/tasks/install.yml new file mode 100644 index 0000000..7678e7d --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/tasks/install.yml @@ -0,0 +1,97 @@ +--- +- name: Cilium | Ensure BPFFS mounted + ansible.posix.mount: + fstype: bpf + path: /sys/fs/bpf + src: bpffs + state: mounted + +- name: Cilium | Create Cilium certs directory + file: + dest: "{{ cilium_cert_dir }}" + state: directory + mode: 0750 + owner: root + group: root + when: + - cilium_identity_allocation_mode == "kvstore" + +- name: Cilium | Link etcd certificates for cilium + file: + src: "{{ etcd_cert_dir }}/{{ item.s }}" + dest: "{{ cilium_cert_dir }}/{{ item.d }}" + mode: 0644 + state: hard + force: yes + loop: + - {s: "{{ kube_etcd_cacert_file }}", d: "ca_cert.crt"} + - {s: "{{ kube_etcd_cert_file }}", d: "cert.crt"} + - {s: "{{ kube_etcd_key_file }}", d: "key.pem"} + when: + - cilium_identity_allocation_mode == "kvstore" + +- name: Cilium | Create hubble dir + file: + path: "{{ kube_config_dir }}/addons/hubble" + state: directory + owner: root + group: root + mode: 0755 + when: + - inventory_hostname == groups['kube_control_plane'][0] + - cilium_hubble_install + +- name: Cilium | Create Cilium node manifests + template: + src: "{{ item.name }}/{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.name }}-{{ item.file }}" + mode: 0644 + loop: + - {name: cilium, file: config.yml, type: cm} + - {name: cilium-operator, file: crb.yml, type: clusterrolebinding} + - {name: cilium-operator, file: cr.yml, type: clusterrole} + - {name: cilium, file: crb.yml, type: clusterrolebinding} + - {name: cilium, file: cr.yml, type: clusterrole} + - {name: cilium, file: secret.yml, type: secret, when: "{{ cilium_encryption_enabled and cilium_encryption_type == 'ipsec' }}"} + - {name: cilium, file: ds.yml, type: ds} + - {name: cilium-operator, file: deploy.yml, type: deploy} + - {name: cilium-operator, file: sa.yml, type: sa} + - {name: cilium, file: sa.yml, type: sa} + register: cilium_node_manifests + when: + - inventory_hostname in groups['kube_control_plane'] + - item.when | default(True) | bool + +- name: Cilium | Create Cilium Hubble manifests + template: + src: "{{ item.name }}/{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/addons/hubble/{{ item.name }}-{{ item.file }}" + mode: 0644 + loop: + - {name: hubble, file: config.yml, type: cm} + - {name: hubble, file: crb.yml, type: clusterrolebinding} + - {name: hubble, file: cr.yml, type: clusterrole} + - {name: hubble, file: cronjob.yml, type: cronjob, when: "{{ cilium_hubble_tls_generate }}"} + - {name: hubble, file: deploy.yml, type: deploy} + - {name: hubble, file: job.yml, type: job, when: "{{ cilium_hubble_tls_generate }}"} + - {name: hubble, file: sa.yml, type: sa} + - {name: hubble, file: service.yml, type: service} + register: cilium_hubble_manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] + - cilium_enable_hubble and cilium_hubble_install + - item.when | default(True) | bool + +- name: Cilium | Enable portmap addon + template: + src: 000-cilium-portmap.conflist.j2 + dest: /etc/cni/net.d/000-cilium-portmap.conflist + mode: 0644 + when: cilium_enable_portmap + +- name: Cilium | Copy Ciliumcli binary from download dir + copy: + src: "{{ downloads.ciliumcli.dest }}" + dest: "{{ bin_dir }}/cilium" + mode: 0755 + remote_src: yes diff --git a/kubespray/roles/network_plugin/cilium/tasks/main.yml b/kubespray/roles/network_plugin/cilium/tasks/main.yml new file mode 100644 index 0000000..8123c5a --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/tasks/main.yml @@ -0,0 +1,9 @@ +--- +- name: Cilium check + import_tasks: check.yml + +- name: Cilium install + include_tasks: install.yml + +- name: Cilium apply + include_tasks: apply.yml diff --git a/kubespray/roles/network_plugin/cilium/tasks/reset.yml b/kubespray/roles/network_plugin/cilium/tasks/reset.yml new file mode 100644 index 0000000..b578b07 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/tasks/reset.yml @@ -0,0 +1,9 @@ +--- +- name: Reset | check and remove devices if still present + include_tasks: reset_iface.yml + vars: + iface: "{{ item }}" + loop: + - cilium_host + - cilium_net + - cilium_vxlan diff --git a/kubespray/roles/network_plugin/cilium/tasks/reset_iface.yml b/kubespray/roles/network_plugin/cilium/tasks/reset_iface.yml new file mode 100644 index 0000000..e2f7c14 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/tasks/reset_iface.yml @@ -0,0 +1,12 @@ +--- +- name: "Reset | check if network device {{ iface }} is present" + stat: + path: "/sys/class/net/{{ iface }}" + get_attributes: no + get_checksum: no + get_mime: no + register: device_remains + +- name: "Reset | remove network device {{ iface }}" + command: "ip link del {{ iface }}" + when: device_remains.stat.exists diff --git a/kubespray/roles/network_plugin/cilium/templates/000-cilium-portmap.conflist.j2 b/kubespray/roles/network_plugin/cilium/templates/000-cilium-portmap.conflist.j2 new file mode 100644 index 0000000..982a7c9 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/000-cilium-portmap.conflist.j2 @@ -0,0 +1,13 @@ +{ + "cniVersion": "0.3.1", + "name": "cilium-portmap", + "plugins": [ + { + "type": "cilium-cni" + }, + { + "type": "portmap", + "capabilities": { "portMappings": true } + } + ] +} diff --git a/kubespray/roles/network_plugin/cilium/templates/cilium-operator/cr.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/cilium-operator/cr.yml.j2 new file mode 100644 index 0000000..642a667 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/cilium-operator/cr.yml.j2 @@ -0,0 +1,169 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cilium-operator +rules: +- apiGroups: + - "" + resources: + # to automatically delete [core|kube]dns pods so that are starting to being + # managed by Cilium + - pods + verbs: + - get + - list + - watch + - delete +- apiGroups: + - "" + resources: + - nodes + verbs: + - list + - watch +- apiGroups: + - "" + resources: + # To remove node taints + - nodes + # To set NetworkUnavailable false on startup + - nodes/status + verbs: + - patch +- apiGroups: + - discovery.k8s.io + resources: + - endpointslices + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + - services + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + # to perform LB IP allocation for BGP + - services/status + verbs: + - update + - patch +- apiGroups: + - "" + resources: + # to perform the translation of a CNP that contains `ToGroup` to its endpoints + - services + - endpoints + # to check apiserver connectivity + - namespaces + verbs: + - get + - list + - watch +- apiGroups: + - cilium.io + resources: + - ciliumnetworkpolicies + - ciliumnetworkpolicies/status + - ciliumnetworkpolicies/finalizers + - ciliumclusterwidenetworkpolicies + - ciliumclusterwidenetworkpolicies/status + - ciliumclusterwidenetworkpolicies/finalizers + - ciliumendpoints + - ciliumendpoints/status + - ciliumendpoints/finalizers + - ciliumnodes + - ciliumnodes/status + - ciliumnodes/finalizers + - ciliumidentities + - ciliumidentities/status + - ciliumidentities/finalizers + - ciliumlocalredirectpolicies + - ciliumlocalredirectpolicies/status + - ciliumlocalredirectpolicies/finalizers +{% if cilium_version | regex_replace('v') is version('1.11', '>=') %} + - ciliumendpointslices +{% endif %} +{% if cilium_version | regex_replace('v') is version('1.12', '>=') %} + - ciliumbgploadbalancerippools + - ciliumloadbalancerippools + - ciliumloadbalancerippools/status + - ciliumbgppeeringpolicies + - ciliumenvoyconfigs +{% endif %} + verbs: + - '*' +- apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + verbs: + - create + - get + - list + - update + - watch +# For cilium-operator running in HA mode. +# +# Cilium operator running in HA mode requires the use of ResourceLock for Leader Election +# between multiple running instances. +# The preferred way of doing this is to use LeasesResourceLock as edits to Leases are less +# common and fewer objects in the cluster watch "all Leases". +- apiGroups: + - coordination.k8s.io + resources: + - leases + verbs: + - create + - get + - update +{% if cilium_version | regex_replace('v') is version('1.12', '>=') %} +- apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + verbs: + - update + resourceNames: + - ciliumbgploadbalancerippools.cilium.io + - ciliumbgppeeringpolicies.cilium.io + - ciliumclusterwideenvoyconfigs.cilium.io + - ciliumclusterwidenetworkpolicies.cilium.io + - ciliumegressgatewaypolicies.cilium.io + - ciliumegressnatpolicies.cilium.io + - ciliumendpoints.cilium.io + - ciliumendpointslices.cilium.io + - ciliumenvoyconfigs.cilium.io + - ciliumexternalworkloads.cilium.io + - ciliumidentities.cilium.io + - ciliumlocalredirectpolicies.cilium.io + - ciliumnetworkpolicies.cilium.io + - ciliumnodes.cilium.io +{% endif %} +{% for rules in cilium_clusterrole_rules_operator_extra_vars %} +- apiGroups: +{% for api in rules['apiGroups'] %} + - {{ api }} +{% endfor %} + resources: +{% for resource in rules['resources'] %} + - {{ resource }} +{% endfor %} + verbs: +{% for verb in rules['verbs'] %} + - {{ verb }} +{% endfor %} +{% if 'resourceNames' in rules %} + resourceNames: +{% for resourceName in rules['resourceNames'] %} + - {{ resourceName }} +{% endfor %} +{% endif %} +{% endfor %} diff --git a/kubespray/roles/network_plugin/cilium/templates/cilium-operator/crb.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/cilium-operator/crb.yml.j2 new file mode 100644 index 0000000..00f0835 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/cilium-operator/crb.yml.j2 @@ -0,0 +1,13 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cilium-operator +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cilium-operator +subjects: +- kind: ServiceAccount + name: cilium-operator + namespace: kube-system diff --git a/kubespray/roles/network_plugin/cilium/templates/cilium-operator/deploy.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/cilium-operator/deploy.yml.j2 new file mode 100644 index 0000000..1418965 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/cilium-operator/deploy.yml.j2 @@ -0,0 +1,170 @@ +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + name: cilium-operator + namespace: kube-system + labels: + io.cilium/app: operator + name: cilium-operator +spec: +{% if groups.k8s_cluster | length == 1 %} + replicas: 1 +{% else %} + replicas: {{ cilium_operator_replicas }} +{% endif %} + selector: + matchLabels: + io.cilium/app: operator + name: cilium-operator + strategy: + rollingUpdate: + maxSurge: 1 + maxUnavailable: 1 + type: RollingUpdate + template: + metadata: +{% if cilium_enable_prometheus %} + annotations: + prometheus.io/port: "{{ cilium_operator_scrape_port }}" + prometheus.io/scrape: "true" +{% endif %} + labels: + io.cilium/app: operator + name: cilium-operator + spec: + containers: + - name: cilium-operator + image: "{{ cilium_operator_image_repo }}:{{ cilium_operator_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + command: + - cilium-operator + args: + - --config-dir=/tmp/cilium/config-map + - --debug=$(CILIUM_DEBUG) +{% if cilium_operator_custom_args is string %} + - {{ cilium_operator_custom_args }} +{% else %} +{% for flag in cilium_operator_custom_args %} + - {{ flag }} +{% endfor %} +{% endif %} + env: + - name: K8S_NODE_NAME + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: spec.nodeName + - name: CILIUM_K8S_NAMESPACE + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + - name: CILIUM_DEBUG + valueFrom: + configMapKeyRef: + key: debug + name: cilium-config + optional: true + - name: AWS_ACCESS_KEY_ID + valueFrom: + secretKeyRef: + name: cilium-aws + key: AWS_ACCESS_KEY_ID + optional: true + - name: AWS_SECRET_ACCESS_KEY + valueFrom: + secretKeyRef: + name: cilium-aws + key: AWS_SECRET_ACCESS_KEY + optional: true + - name: AWS_DEFAULT_REGION + valueFrom: + secretKeyRef: + name: cilium-aws + key: AWS_DEFAULT_REGION + optional: true +{% if cilium_kube_proxy_replacement == 'strict' %} + - name: KUBERNETES_SERVICE_HOST + value: "{{ kube_apiserver_global_endpoint | urlsplit('hostname') }}" + - name: KUBERNETES_SERVICE_PORT + value: "{{ kube_apiserver_global_endpoint | urlsplit('port') }}" +{% endif %} +{% if cilium_enable_prometheus %} + - name: POD_NAMESPACE + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + ports: + - name: prometheus + containerPort: {{ cilium_operator_scrape_port }} + hostPort: {{ cilium_operator_scrape_port }} + protocol: TCP +{% endif %} + livenessProbe: + httpGet: +{% if cilium_enable_ipv4 %} + host: 127.0.0.1 +{% else %} + host: '::1' +{% endif %} + path: /healthz + port: 9234 + scheme: HTTP + initialDelaySeconds: 60 + periodSeconds: 10 + timeoutSeconds: 3 + volumeMounts: + - name: cilium-config-path + mountPath: /tmp/cilium/config-map + readOnly: true +{% if cilium_identity_allocation_mode == "kvstore" %} + - name: etcd-config-path + mountPath: /var/lib/etcd-config + readOnly: true + - name: etcd-secrets + mountPath: "{{ cilium_cert_dir }}" + readOnly: true +{% endif %} +{% for volume_mount in cilium_operator_extra_volume_mounts %} + - {{ volume_mount | to_nice_yaml(indent=2) | indent(14) }} +{% endfor %} + hostNetwork: true + dnsPolicy: ClusterFirstWithHostNet + restartPolicy: Always + priorityClassName: system-node-critical + serviceAccount: cilium-operator + serviceAccountName: cilium-operator + # In HA mode, cilium-operator pods must not be scheduled on the same + # node as they will clash with each other. + affinity: + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - topologyKey: kubernetes.io/hostname + labelSelector: + matchLabels: + io.cilium/app: operator + tolerations: + - operator: Exists + volumes: + - name: cilium-config-path + configMap: + name: cilium-config +{% if cilium_identity_allocation_mode == "kvstore" %} + # To read the etcd config stored in config maps + - name: etcd-config-path + configMap: + name: cilium-config + defaultMode: 420 + items: + - key: etcd-config + path: etcd.config + # To read the k8s etcd secrets in case the user might want to use TLS + - name: etcd-secrets + hostPath: + path: "{{ cilium_cert_dir }}" +{% endif %} +{% for volume in cilium_operator_extra_volumes %} + - {{ volume | to_nice_yaml(indent=2) | indent(10) }} +{% endfor %} diff --git a/kubespray/roles/network_plugin/cilium/templates/cilium-operator/sa.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/cilium-operator/sa.yml.j2 new file mode 100644 index 0000000..c5d1893 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/cilium-operator/sa.yml.j2 @@ -0,0 +1,6 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: cilium-operator + namespace: kube-system diff --git a/kubespray/roles/network_plugin/cilium/templates/cilium/config.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/cilium/config.yml.j2 new file mode 100644 index 0000000..399d8ce --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/cilium/config.yml.j2 @@ -0,0 +1,256 @@ +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: cilium-config + namespace: kube-system +data: + identity-allocation-mode: {{ cilium_identity_allocation_mode }} + +{% if cilium_identity_allocation_mode == "kvstore" %} + # This etcd-config contains the etcd endpoints of your cluster. If you use + # TLS please make sure you follow the tutorial in https://cilium.link/etcd-config + etcd-config: |- + --- + endpoints: +{% for ip_addr in etcd_access_addresses.split(',') %} + - {{ ip_addr }} +{% endfor %} + + # In case you want to use TLS in etcd, uncomment the 'ca-file' line + # and create a kubernetes secret by following the tutorial in + # https://cilium.link/etcd-config + ca-file: "{{ cilium_cert_dir }}/ca_cert.crt" + + # In case you want client to server authentication, uncomment the following + # lines and create a kubernetes secret by following the tutorial in + # https://cilium.link/etcd-config + key-file: "{{ cilium_cert_dir }}/key.pem" + cert-file: "{{ cilium_cert_dir }}/cert.crt" + + # kvstore + # https://docs.cilium.io/en/latest/cmdref/kvstore/ + kvstore: etcd + kvstore-opt: '{"etcd.config": "/var/lib/etcd-config/etcd.config"}' +{% endif %} + + # If you want metrics enabled in all of your Cilium agents, set the port for + # which the Cilium agents will have their metrics exposed. + # This option deprecates the "prometheus-serve-addr" in the + # "cilium-metrics-config" ConfigMap + # NOTE that this will open the port on ALL nodes where Cilium pods are + # scheduled. +{% if cilium_enable_prometheus %} + prometheus-serve-addr: ":{{ cilium_agent_scrape_port }}" + operator-prometheus-serve-addr: ":{{ cilium_operator_scrape_port }}" + enable-metrics: "true" +{% endif %} + + # If you want to run cilium in debug mode change this value to true + debug: "{{ cilium_debug }}" + enable-ipv4: "{{ cilium_enable_ipv4 }}" + enable-ipv6: "{{ cilium_enable_ipv6 }}" + # If a serious issue occurs during Cilium startup, this + # invasive option may be set to true to remove all persistent + # state. Endpoints will not be restored using knowledge from a + # prior Cilium run, so they may receive new IP addresses upon + # restart. This also triggers clean-cilium-bpf-state. + clean-cilium-state: "false" + # If you want to clean cilium BPF state, set this to true; + # Removes all BPF maps from the filesystem. Upon restart, + # endpoints are restored with the same IP addresses, however + # any ongoing connections may be disrupted briefly. + # Loadbalancing decisions will be reset, so any ongoing + # connections via a service may be loadbalanced to a different + # backend after restart. + clean-cilium-bpf-state: "false" + + # Users who wish to specify their own custom CNI configuration file must set + # custom-cni-conf to "true", otherwise Cilium may overwrite the configuration. + custom-cni-conf: "false" + + # If you want cilium monitor to aggregate tracing for packets, set this level + # to "low", "medium", or "maximum". The higher the level, the less packets + # that will be seen in monitor output. + monitor-aggregation: "{{ cilium_monitor_aggregation }}" + + # ct-global-max-entries-* specifies the maximum number of connections + # supported across all endpoints, split by protocol: tcp or other. One pair + # of maps uses these values for IPv4 connections, and another pair of maps + # use these values for IPv6 connections. + # + # If these values are modified, then during the next Cilium startup the + # tracking of ongoing connections may be disrupted. This may lead to brief + # policy drops or a change in loadbalancing decisions for a connection. + # + # For users upgrading from Cilium 1.2 or earlier, to minimize disruption + # during the upgrade process, comment out these options. + bpf-ct-global-tcp-max: "524288" + bpf-ct-global-any-max: "262144" + + # Pre-allocation of map entries allows per-packet latency to be reduced, at + # the expense of up-front memory allocation for the entries in the maps. The + # default value below will minimize memory usage in the default installation; + # users who are sensitive to latency may consider setting this to "true". + # + # This option was introduced in Cilium 1.4. Cilium 1.3 and earlier ignore + # this option and behave as though it is set to "true". + # + # If this value is modified, then during the next Cilium startup the restore + # of existing endpoints and tracking of ongoing connections may be disrupted. + # This may lead to policy drops or a change in loadbalancing decisions for a + # connection for some time. Endpoints may need to be recreated to restore + # connectivity. + # + # If this option is set to "false" during an upgrade from 1.3 or earlier to + # 1.4 or later, then it may cause one-time disruptions during the upgrade. + preallocate-bpf-maps: "{{ cilium_preallocate_bpf_maps }}" + + # Regular expression matching compatible Istio sidecar istio-proxy + # container image names + sidecar-istio-proxy-image: "cilium/istio_proxy" + + # Encapsulation mode for communication between nodes + # Possible values: + # - disabled + # - vxlan (default) + # - geneve + tunnel: "{{ cilium_tunnel_mode }}" + + # Enable Bandwidth Manager + # Cilium’s bandwidth manager supports the kubernetes.io/egress-bandwidth Pod annotation. + # Bandwidth enforcement currently does not work in combination with L7 Cilium Network Policies. + # In case they select the Pod at egress, then the bandwidth enforcement will be disabled for those Pods. + # Bandwidth Manager requires a v5.1.x or more recent Linux kernel. +{% if cilium_enable_bandwidth_manager %} + enable-bandwidth-manager: "true" +{% endif %} + + # Name of the cluster. Only relevant when building a mesh of clusters. + cluster-name: "{{ cilium_cluster_name }}" + + # Unique ID of the cluster. Must be unique across all conneted clusters and + # in the range of 1 and 255. Only relevant when building a mesh of clusters. + #cluster-id: 1 +{% if cilium_cluster_id is defined %} + cluster-id: "{{ cilium_cluster_id }}" +{% endif %} + +# `wait-bpf-mount` is removed after v1.10.4 +# https://github.com/cilium/cilium/commit/d2217045cb3726a7f823174e086913b69b8090da +{% if cilium_version | regex_replace('v') is version('1.10.4', '<') %} + # wait-bpf-mount makes init container wait until bpf filesystem is mounted + wait-bpf-mount: "false" +{% endif %} + + kube-proxy-replacement: "{{ cilium_kube_proxy_replacement }}" + +# `native-routing-cidr` is deprecated in 1.10, removed in 1.12. +# Replaced by `ipv4-native-routing-cidr` +# https://github.com/cilium/cilium/pull/16695 +{% if cilium_version | regex_replace('v') is version('1.12', '<') %} + native-routing-cidr: "{{ cilium_native_routing_cidr }}" +{% else %} +{% if cilium_native_routing_cidr | length %} + ipv4-native-routing-cidr: "{{ cilium_native_routing_cidr }}" +{% endif %} +{% if cilium_native_routing_cidr_ipv6 | length %} + ipv6-native-routing-cidr: "{{ cilium_native_routing_cidr_ipv6 }}" +{% endif %} +{% endif %} + + auto-direct-node-routes: "{{ cilium_auto_direct_node_routes }}" + + operator-api-serve-addr: "{{ cilium_operator_api_serve_addr }}" + + # Hubble settings +{% if cilium_enable_hubble %} + enable-hubble: "true" +{% if cilium_enable_hubble_metrics %} + hubble-metrics-server: ":{{ cilium_hubble_scrape_port }}" + hubble-metrics: +{% for hubble_metrics_cycle in cilium_hubble_metrics %} + {{ hubble_metrics_cycle }} +{% endfor %} +{% endif %} + hubble-listen-address: ":4244" +{% if cilium_enable_hubble and cilium_hubble_install %} + hubble-disable-tls: "{% if cilium_hubble_tls_generate %}false{% else %}true{% endif %}" + hubble-tls-cert-file: /var/lib/cilium/tls/hubble/server.crt + hubble-tls-key-file: /var/lib/cilium/tls/hubble/server.key + hubble-tls-client-ca-files: /var/lib/cilium/tls/hubble/client-ca.crt +{% endif %} +{% endif %} + + # IP Masquerade Agent + enable-ip-masq-agent: "{{ cilium_ip_masq_agent_enable }}" + +{% for key, value in cilium_config_extra_vars.items() %} + {{ key }}: "{{ value }}" +{% endfor %} + + # Enable transparent network encryption +{% if cilium_encryption_enabled %} +{% if cilium_encryption_type == "ipsec" %} + enable-ipsec: "true" + ipsec-key-file: /etc/ipsec/keys + encrypt-node: "{{ cilium_ipsec_node_encryption }}" +{% endif %} + +{% if cilium_encryption_type == "wireguard" %} + enable-wireguard: "true" + enable-wireguard-userspace-fallback: "{{ cilium_wireguard_userspace_fallback }}" +{% endif %} +{% endif %} + + # IPAM settings + ipam: "{{ cilium_ipam_mode }}" +{% if cilium_ipam_mode == "cluster-pool" %} + cluster-pool-ipv4-cidr: "{{ cilium_pool_cidr | default(kube_pods_subnet) }}" + cluster-pool-ipv4-mask-size: "{{ cilium_pool_mask_size | default(kube_network_node_prefix) }}" +{% if cilium_enable_ipv6 %} + cluster-pool-ipv6-cidr: "{{ cilium_pool_cidr_ipv6 | default(kube_pods_subnet_ipv6) }}" + cluster-pool-ipv6-mask-size: "{{ cilium_pool_mask_size_ipv6 | default(kube_network_node_prefix_ipv6) }}" +{% endif %} +{% endif %} + + agent-health-port: "{{ cilium_agent_health_port }}" + +{% if cilium_version | regex_replace('v') is version('1.11', '>=') and cilium_cgroup_host_root != '' %} + cgroup-root: "{{ cilium_cgroup_host_root }}" +{% endif %} + + bpf-map-dynamic-size-ratio: "{{ cilium_bpf_map_dynamic_size_ratio }}" + + enable-ipv4-masquerade: "{{ cilium_enable_ipv4_masquerade }}" + enable-ipv6-masquerade: "{{ cilium_enable_ipv6_masquerade }}" + + enable-bpf-masquerade: "{{ cilium_enable_bpf_masquerade }}" + + enable-host-legacy-routing: "{{ cilium_enable_host_legacy_routing }}" + + enable-remote-node-identity: "{{ cilium_enable_remote_node_identity }}" + + enable-well-known-identities: "{{ cilium_enable_well_known_identities }}" + + monitor-aggregation-flags: "{{ cilium_monitor_aggregation_flags }}" + + enable-bpf-clock-probe: "{{ cilium_enable_bpf_clock_probe }}" + + disable-cnp-status-updates: "{{ cilium_disable_cnp_status_updates }}" +{% if cilium_ip_masq_agent_enable %} +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: ip-masq-agent + namespace: kube-system +data: + config: | + nonMasqueradeCIDRs: +{% for cidr in cilium_non_masquerade_cidrs %} + - {{ cidr }} +{% endfor %} + masqLinkLocal: {{ cilium_masq_link_local | bool }} + resyncInterval: "{{ cilium_ip_masq_resync_interval }}" +{% endif %} diff --git a/kubespray/roles/network_plugin/cilium/templates/cilium/cr.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/cilium/cr.yml.j2 new file mode 100644 index 0000000..a16211c --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/cilium/cr.yml.j2 @@ -0,0 +1,122 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cilium +rules: +- apiGroups: + - networking.k8s.io + resources: + - networkpolicies + verbs: + - get + - list + - watch +- apiGroups: + - discovery.k8s.io + resources: + - endpointslices + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + - namespaces + - services + - pods + - endpoints + - nodes + verbs: + - get + - list + - watch +{% if cilium_version | regex_replace('v') is version('1.12', '<') %} +- apiGroups: + - "" + resources: + - pods + - pods/finalizers + verbs: + - get + - list + - watch + - update + - delete +- apiGroups: + - "" + resources: + - pods + - nodes + verbs: + - get + - list + - watch + - update +{% endif %} +- apiGroups: + - "" + resources: + - nodes + - nodes/status + verbs: + - patch +- apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + verbs: + # Deprecated for removal in v1.10 + - create + - list + - watch + - update + + # This is used when validating policies in preflight. This will need to stay + # until we figure out how to avoid "get" inside the preflight, and then + # should be removed ideally. + - get +- apiGroups: + - cilium.io + resources: + - ciliumnetworkpolicies + - ciliumnetworkpolicies/status + - ciliumclusterwidenetworkpolicies + - ciliumclusterwidenetworkpolicies/status + - ciliumendpoints + - ciliumendpoints/status + - ciliumnodes + - ciliumnodes/status + - ciliumidentities + - ciliumlocalredirectpolicies + - ciliumlocalredirectpolicies/status + - ciliumegressnatpolicies +{% if cilium_version | regex_replace('v') is version('1.11', '>=') %} + - ciliumendpointslices +{% endif %} +{% if cilium_version | regex_replace('v') is version('1.12', '>=') %} + - ciliumbgploadbalancerippools + - ciliumbgppeeringpolicies +{% endif %} +{% if cilium_version | regex_replace('v') is version('1.11.5', '<') %} + - ciliumnetworkpolicies/finalizers + - ciliumclusterwidenetworkpolicies/finalizers + - ciliumendpoints/finalizers + - ciliumnodes/finalizers + - ciliumidentities/finalizers + - ciliumlocalredirectpolicies/finalizers +{% endif %} + verbs: + - '*' +{% if cilium_version | regex_replace('v') is version('1.12', '>=') %} +- apiGroups: + - cilium.io + resources: + - ciliumclusterwideenvoyconfigs + - ciliumenvoyconfigs + - ciliumegressgatewaypolicies + verbs: + - list + - watch +{% endif %} diff --git a/kubespray/roles/network_plugin/cilium/templates/cilium/crb.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/cilium/crb.yml.j2 new file mode 100644 index 0000000..d23897f --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/cilium/crb.yml.j2 @@ -0,0 +1,13 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cilium +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cilium +subjects: +- kind: ServiceAccount + name: cilium + namespace: kube-system diff --git a/kubespray/roles/network_plugin/cilium/templates/cilium/ds.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/cilium/ds.yml.j2 new file mode 100644 index 0000000..3836034 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/cilium/ds.yml.j2 @@ -0,0 +1,444 @@ +--- +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: cilium + namespace: kube-system + labels: + k8s-app: cilium +spec: + selector: + matchLabels: + k8s-app: cilium + updateStrategy: + rollingUpdate: + # Specifies the maximum number of Pods that can be unavailable during the update process. + maxUnavailable: 2 + type: RollingUpdate + template: + metadata: + annotations: +{% if cilium_enable_prometheus %} + prometheus.io/port: "{{ cilium_agent_scrape_port }}" + prometheus.io/scrape: "true" +{% endif %} + scheduler.alpha.kubernetes.io/tolerations: '[{"key":"dedicated","operator":"Equal","value":"master","effect":"NoSchedule"}]' + labels: + k8s-app: cilium + spec: + containers: + - name: cilium-agent + image: "{{ cilium_image_repo }}:{{ cilium_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + command: + - cilium-agent + args: + - --config-dir=/tmp/cilium/config-map +{% if cilium_mtu != "" %} + - --mtu={{ cilium_mtu }} +{% endif %} +{% if cilium_agent_custom_args is string %} + - {{ cilium_agent_custom_args }} +{% else %} +{% for flag in cilium_agent_custom_args %} + - {{ flag }} +{% endfor %} +{% endif %} + startupProbe: + httpGet: + host: '127.0.0.1' + path: /healthz + port: {{ cilium_agent_health_port }} + scheme: HTTP + httpHeaders: + - name: "brief" + value: "true" + failureThreshold: 105 + periodSeconds: 2 + successThreshold: 1 + livenessProbe: + httpGet: + host: '127.0.0.1' + path: /healthz + port: {{ cilium_agent_health_port }} + scheme: HTTP + httpHeaders: + - name: "brief" + value: "true" + failureThreshold: 10 + periodSeconds: 30 + successThreshold: 1 + timeoutSeconds: 5 + readinessProbe: + httpGet: + host: 127.0.0.1 + path: /healthz + port: {{ cilium_agent_health_port }} + scheme: HTTP + httpHeaders: + - name: "brief" + value: "true" + initialDelaySeconds: 5 + periodSeconds: 30 + successThreshold: 1 + failureThreshold: 3 + timeoutSeconds: 5 + env: + - name: K8S_NODE_NAME + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: spec.nodeName + - name: CILIUM_K8S_NAMESPACE + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + - name: CILIUM_CLUSTERMESH_CONFIG + value: /var/lib/cilium/clustermesh/ +{% if cilium_kube_proxy_replacement == 'strict' %} + - name: KUBERNETES_SERVICE_HOST + value: "{{ kube_apiserver_global_endpoint | urlsplit('hostname') }}" + - name: KUBERNETES_SERVICE_PORT + value: "{{ kube_apiserver_global_endpoint | urlsplit('port') }}" +{% endif %} +{% for env_var in cilium_agent_extra_env_vars %} + - {{ env_var | to_nice_yaml(indent=2) | indent(10) }} +{% endfor %} + lifecycle: + postStart: + exec: + command: + - "/cni-install.sh" + - "--cni-exclusive={{ cilium_cni_exclusive | string | lower }}" +{% if cilium_version | regex_replace('v') is version('1.12', '>=') %} + - "--enable-debug={{ cilium_debug | string | lower }}" + - "--log-file={{ cilium_cni_log_file }}" +{% endif %} + preStop: + exec: + command: + - /cni-uninstall.sh + resources: + limits: + cpu: {{ cilium_cpu_limit }} + memory: {{ cilium_memory_limit }} + requests: + cpu: {{ cilium_cpu_requests }} + memory: {{ cilium_memory_requests }} +{% if cilium_enable_prometheus or cilium_enable_hubble_metrics %} + ports: +{% endif %} +{% if cilium_enable_prometheus %} + - name: prometheus + containerPort: {{ cilium_agent_scrape_port }} + hostPort: {{ cilium_agent_scrape_port }} + protocol: TCP +{% endif %} +{% if cilium_enable_hubble_metrics %} + - name: hubble-metrics + containerPort: {{ cilium_hubble_scrape_port }} + hostPort: {{ cilium_hubble_scrape_port }} + protocol: TCP +{% endif %} + securityContext: + privileged: true + volumeMounts: + - name: bpf-maps + mountPath: /sys/fs/bpf + mountPropagation: Bidirectional + - name: cilium-run + mountPath: /var/run/cilium +{% if cilium_version | regex_replace('v') is version('1.13.1', '<') %} + - name: cni-path + mountPath: /host/opt/cni/bin +{% endif %} + - name: etc-cni-netd + mountPath: /host/etc/cni/net.d +{% if cilium_identity_allocation_mode == "kvstore" %} + - name: etcd-config-path + mountPath: /var/lib/etcd-config + readOnly: true + - name: etcd-secrets + mountPath: "{{ cilium_cert_dir }}" + readOnly: true +{% endif %} + - name: clustermesh-secrets + mountPath: /var/lib/cilium/clustermesh + readOnly: true + - name: cilium-config-path + mountPath: /tmp/cilium/config-map + readOnly: true +{% if cilium_ip_masq_agent_enable %} + - name: ip-masq-agent + mountPath: /etc/config + readOnly: true +{% endif %} + # Needed to be able to load kernel modules + - name: lib-modules + mountPath: /lib/modules + readOnly: true + - name: xtables-lock + mountPath: /run/xtables.lock +{% if cilium_encryption_enabled and cilium_encryption_type == "ipsec" %} + - name: cilium-ipsec-secrets + mountPath: /etc/ipsec + readOnly: true +{% endif %} +{% if cilium_hubble_install %} + - name: hubble-tls + mountPath: /var/lib/cilium/tls/hubble + readOnly: true +{% endif %} +{% for volume_mount in cilium_agent_extra_volume_mounts %} + - {{ volume_mount | to_nice_yaml(indent=2) | indent(10) }} +{% endfor %} +# In managed etcd mode, Cilium must be able to resolve the DNS name of the etcd service +{% if cilium_identity_allocation_mode == "kvstore" %} + dnsPolicy: ClusterFirstWithHostNet +{% endif %} + hostNetwork: true + initContainers: +{% if cilium_version | regex_replace('v') is version('1.11', '>=') and cilium_cgroup_auto_mount %} + - name: mount-cgroup + image: "{{ cilium_image_repo }}:{{ cilium_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + env: + - name: CGROUP_ROOT + value: {{ cilium_cgroup_host_root }} + - name: BIN_PATH + value: /opt/cni/bin + command: + - sh + - -ec + # The statically linked Go program binary is invoked to avoid any + # dependency on utilities like sh and mount that can be missing on certain + # distros installed on the underlying host. Copy the binary to the + # same directory where we install cilium cni plugin so that exec permissions + # are available. + - | + cp /usr/bin/cilium-mount /hostbin/cilium-mount; + nsenter --cgroup=/hostproc/1/ns/cgroup --mount=/hostproc/1/ns/mnt "${BIN_PATH}/cilium-mount" $CGROUP_ROOT; + rm /hostbin/cilium-mount + volumeMounts: + - name: hostproc + mountPath: /hostproc + - name: cni-path + mountPath: /hostbin + securityContext: + privileged: true +{% endif %} +{% if cilium_version | regex_replace('v') is version('1.11.7', '>=') %} + - name: apply-sysctl-overwrites + image: "{{ cilium_image_repo }}:{{ cilium_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + env: + - name: BIN_PATH + value: /opt/cni/bin + command: + - sh + - -ec + # The statically linked Go program binary is invoked to avoid any + # dependency on utilities like sh that can be missing on certain + # distros installed on the underlying host. Copy the binary to the + # same directory where we install cilium cni plugin so that exec permissions + # are available. + - | + cp /usr/bin/cilium-sysctlfix /hostbin/cilium-sysctlfix; + nsenter --mount=/hostproc/1/ns/mnt "${BIN_PATH}/cilium-sysctlfix"; + rm /hostbin/cilium-sysctlfix + volumeMounts: + - name: hostproc + mountPath: /hostproc + - name: cni-path + mountPath: /hostbin + securityContext: + privileged: true +{% endif %} + - name: clean-cilium-state + image: "{{ cilium_image_repo }}:{{ cilium_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + command: + - /init-container.sh + env: + - name: CILIUM_ALL_STATE + valueFrom: + configMapKeyRef: + name: cilium-config + key: clean-cilium-state + optional: true + - name: CILIUM_BPF_STATE + valueFrom: + configMapKeyRef: + name: cilium-config + key: clean-cilium-bpf-state + optional: true +# Removed in 1.11 and up. +# https://github.com/cilium/cilium/commit/f7a3f59fd74983c600bfce9cac364b76d20849d9 +{% if cilium_version | regex_replace('v') is version('1.11', '<') %} + - name: CILIUM_WAIT_BPF_MOUNT + valueFrom: + configMapKeyRef: + key: wait-bpf-mount + name: cilium-config + optional: true +{% endif %} +{% if cilium_kube_proxy_replacement == 'strict' %} + - name: KUBERNETES_SERVICE_HOST + value: "{{ kube_apiserver_global_endpoint | urlsplit('hostname') }}" + - name: KUBERNETES_SERVICE_PORT + value: "{{ kube_apiserver_global_endpoint | urlsplit('port') }}" +{% endif %} + securityContext: + privileged: true + volumeMounts: + - name: bpf-maps + mountPath: /sys/fs/bpf +{% if cilium_version | regex_replace('v') is version('1.11', '>=') %} + # Required to mount cgroup filesystem from the host to cilium agent pod + - name: cilium-cgroup + mountPath: {{ cilium_cgroup_host_root }} + mountPropagation: HostToContainer +{% endif %} + - name: cilium-run + mountPath: /var/run/cilium + resources: + requests: + cpu: 100m + memory: 100Mi +{% if cilium_version | regex_replace('v') is version('1.13.1', '>=') %} + # Install the CNI binaries in an InitContainer so we don't have a writable host mount in the agent + - name: install-cni-binaries + image: "{{ cilium_image_repo }}:{{ cilium_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + command: + - "/install-plugin.sh" + resources: + requests: + cpu: 100m + memory: 10Mi + securityContext: + privileged: true + terminationMessagePolicy: FallbackToLogsOnError + volumeMounts: + - name: cni-path + mountPath: /host/opt/cni/bin +{% endif %} + restartPolicy: Always + priorityClassName: system-node-critical + serviceAccount: cilium + serviceAccountName: cilium + terminationGracePeriodSeconds: 1 + hostNetwork: true +# In managed etcd mode, Cilium must be able to resolve the DNS name of the etcd service +{% if cilium_identity_allocation_mode == "kvstore" %} + dnsPolicy: ClusterFirstWithHostNet +{% endif %} + affinity: + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - topologyKey: kubernetes.io/hostname + labelSelector: + matchLabels: + k8s-app: cilium + tolerations: + - operator: Exists + volumes: + # To keep state between restarts / upgrades + - name: cilium-run + hostPath: + path: /var/run/cilium + type: DirectoryOrCreate + # To keep state between restarts / upgrades for bpf maps + - name: bpf-maps + hostPath: + path: /sys/fs/bpf + type: DirectoryOrCreate +{% if cilium_version | regex_replace('v') is version('1.11', '>=') %} + # To mount cgroup2 filesystem on the host + - name: hostproc + hostPath: + path: /proc + type: Directory + # To keep state between restarts / upgrades for cgroup2 filesystem + - name: cilium-cgroup + hostPath: + path: {{ cilium_cgroup_host_root }} + type: DirectoryOrCreate +{% endif %} + # To install cilium cni plugin in the host + - name: cni-path + hostPath: + path: /opt/cni/bin + type: DirectoryOrCreate + # To install cilium cni configuration in the host + - name: etc-cni-netd + hostPath: + path: /etc/cni/net.d + type: DirectoryOrCreate + # To be able to load kernel modules + - name: lib-modules + hostPath: + path: /lib/modules + # To access iptables concurrently with other processes (e.g. kube-proxy) + - name: xtables-lock + hostPath: + path: /run/xtables.lock + type: FileOrCreate +{% if cilium_identity_allocation_mode == "kvstore" %} + # To read the etcd config stored in config maps + - name: etcd-config-path + configMap: + name: cilium-config + # note: the leading zero means this number is in octal representation: do not remove it + defaultMode: 0400 + items: + - key: etcd-config + path: etcd.config + # To read the k8s etcd secrets in case the user might want to use TLS + - name: etcd-secrets + hostPath: + path: "{{ cilium_cert_dir }}" +{% endif %} + # To read the clustermesh configuration + - name: clustermesh-secrets + secret: + secretName: cilium-clustermesh + # note: the leading zero means this number is in octal representation: do not remove it + defaultMode: 0400 + optional: true + # To read the configuration from the config map + - name: cilium-config-path + configMap: + name: cilium-config +{% if cilium_ip_masq_agent_enable %} + - name: ip-masq-agent + configMap: + name: ip-masq-agent + optional: true + items: + - key: config + path: ip-masq-agent +{% endif %} +{% if cilium_encryption_enabled and cilium_encryption_type == "ipsec" %} + - name: cilium-ipsec-secrets + secret: + secretName: cilium-ipsec-keys +{% endif %} +{% if cilium_hubble_install %} + - name: hubble-tls + projected: + # note: the leading zero means this number is in octal representation: do not remove it + defaultMode: 0400 + sources: + - secret: + name: hubble-server-certs + optional: true + items: + - key: ca.crt + path: client-ca.crt + - key: tls.crt + path: server.crt + - key: tls.key + path: server.key +{% endif %} diff --git a/kubespray/roles/network_plugin/cilium/templates/cilium/sa.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/cilium/sa.yml.j2 new file mode 100644 index 0000000..c03ac59 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/cilium/sa.yml.j2 @@ -0,0 +1,6 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: cilium + namespace: kube-system diff --git a/kubespray/roles/network_plugin/cilium/templates/cilium/secret.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/cilium/secret.yml.j2 new file mode 100644 index 0000000..776c689 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/cilium/secret.yml.j2 @@ -0,0 +1,9 @@ +--- +apiVersion: v1 +data: + keys: {{ cilium_ipsec_key }} +kind: Secret +metadata: + name: cilium-ipsec-keys + namespace: kube-system +type: Opaque diff --git a/kubespray/roles/network_plugin/cilium/templates/hubble/config.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/hubble/config.yml.j2 new file mode 100644 index 0000000..c045b43 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/hubble/config.yml.j2 @@ -0,0 +1,70 @@ +#jinja2: trim_blocks:False +--- +# Source: cilium helm chart: cilium/templates/hubble-relay/configmap.yaml +apiVersion: v1 +kind: ConfigMap +metadata: + name: hubble-relay-config + namespace: kube-system +data: + config.yaml: | + peer-service: "hubble-peer.kube-system.svc.{{ dns_domain }}:443" + listen-address: :4245 + metrics-listen-address: ":9966" + dial-timeout: + retry-timeout: + sort-buffer-len-max: + sort-buffer-drain-timeout: + tls-client-cert-file: /var/lib/hubble-relay/tls/client.crt + tls-client-key-file: /var/lib/hubble-relay/tls/client.key + tls-server-cert-file: /var/lib/hubble-relay/tls/server.crt + tls-server-key-file: /var/lib/hubble-relay/tls/server.key + tls-hubble-server-ca-files: /var/lib/hubble-relay/tls/hubble-server-ca.crt + disable-server-tls: {% if cilium_hubble_tls_generate %}false{% else %}true{% endif %} + disable-client-tls: {% if cilium_hubble_tls_generate %}false{% else %}true{% endif %} +--- +# Source: cilium/templates/hubble-ui/configmap.yaml +apiVersion: v1 +kind: ConfigMap +metadata: + name: hubble-ui-nginx + namespace: kube-system +data: + nginx.conf: | + server { + listen 8081; + {% if cilium_enable_ipv6 %} + listen [::]:8081; + {% endif %} + server_name localhost; + root /app; + index index.html; + client_max_body_size 1G; + + location / { + proxy_set_header Host $host; + proxy_set_header X-Real-IP $remote_addr; + + # CORS + add_header Access-Control-Allow-Methods "GET, POST, PUT, HEAD, DELETE, OPTIONS"; + add_header Access-Control-Allow-Origin *; + add_header Access-Control-Max-Age 1728000; + add_header Access-Control-Expose-Headers content-length,grpc-status,grpc-message; + add_header Access-Control-Allow-Headers range,keep-alive,user-agent,cache-control,content-type,content-transfer-encoding,x-accept-content-transfer-encoding,x-accept-response-streaming,x-user-agent,x-grpc-web,grpc-timeout; + if ($request_method = OPTIONS) { + return 204; + } + # /CORS + + location /api { + proxy_http_version 1.1; + proxy_pass_request_headers on; + proxy_hide_header Access-Control-Allow-Origin; + proxy_pass http://127.0.0.1:8090; + } + + location / { + try_files $uri $uri/ /index.html; + } + } + } diff --git a/kubespray/roles/network_plugin/cilium/templates/hubble/cr.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/hubble/cr.yml.j2 new file mode 100644 index 0000000..4a95565 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/hubble/cr.yml.j2 @@ -0,0 +1,106 @@ +{% if cilium_hubble_tls_generate %} +--- +# Source: cilium/templates/hubble-generate-certs-clusterrole.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: hubble-generate-certs +rules: + - apiGroups: + - "" + resources: + - secrets + - configmaps + verbs: + - create + - apiGroups: + - "" + resources: + - secrets + resourceNames: + - hubble-server-certs + - hubble-relay-client-certs + - hubble-relay-server-certs + verbs: + - update + - apiGroups: + - "" + resources: + - configmaps + resourceNames: + - hubble-ca-cert + verbs: + - update + - apiGroups: + - "" + resources: + - secrets + resourceNames: + - hubble-ca-secret + verbs: + - get +{% endif %} +--- +# Source: cilium/templates/hubble-relay-clusterrole.yaml +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: hubble-relay +rules: + - apiGroups: + - "" + resources: + - componentstatuses + - endpoints + - namespaces + - nodes + - pods + - services + verbs: + - get + - list + - watch +--- +# Source: cilium/templates/hubble-ui-clusterrole.yaml +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: hubble-ui +rules: + - apiGroups: + - networking.k8s.io + resources: + - networkpolicies + verbs: + - get + - list + - watch + - apiGroups: + - "" + resources: + - componentstatuses + - endpoints + - namespaces + - nodes + - pods + - services + verbs: + - get + - list + - watch + - apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + verbs: + - get + - list + - watch + - apiGroups: + - cilium.io + resources: + - "*" + verbs: + - get + - list + - watch diff --git a/kubespray/roles/network_plugin/cilium/templates/hubble/crb.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/hubble/crb.yml.j2 new file mode 100644 index 0000000..f033429 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/hubble/crb.yml.j2 @@ -0,0 +1,44 @@ +{% if cilium_hubble_tls_generate %} +--- +# Source: cilium/templates/hubble-generate-certs-clusterrolebinding.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: hubble-generate-certs +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: hubble-generate-certs +subjects: +- kind: ServiceAccount + name: hubble-generate-certs + namespace: kube-system +{% endif %} +--- +# Source: cilium/templates/hubble-relay-clusterrolebinding.yaml +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: hubble-relay +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: hubble-relay +subjects: +- kind: ServiceAccount + namespace: kube-system + name: hubble-relay +--- +# Source: cilium/templates/hubble-ui-clusterrolebinding.yaml +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: hubble-ui +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: hubble-ui +subjects: +- kind: ServiceAccount + namespace: kube-system + name: hubble-ui diff --git a/kubespray/roles/network_plugin/cilium/templates/hubble/cronjob.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/hubble/cronjob.yml.j2 new file mode 100644 index 0000000..8010c52 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/hubble/cronjob.yml.j2 @@ -0,0 +1,38 @@ +--- +# Source: cilium/templates/hubble-generate-certs-cronjob.yaml +apiVersion: batch/v1 +kind: CronJob +metadata: + name: hubble-generate-certs + namespace: kube-system + labels: + k8s-app: hubble-generate-certs +spec: + schedule: "0 0 1 */4 *" + concurrencyPolicy: Forbid + jobTemplate: + spec: + template: + metadata: + labels: + k8s-app: hubble-generate-certs + spec: + serviceAccount: hubble-generate-certs + serviceAccountName: hubble-generate-certs + containers: + - name: certgen + image: "{{ cilium_hubble_certgen_image_repo }}:{{ cilium_hubble_certgen_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + command: + - "/usr/bin/cilium-certgen" + # Because this is executed as a job, we pass the values as command + # line args instead of via config map. This allows users to inspect + # the values used in past runs by inspecting the completed pod. + args: + {% for key, value in cilium_certgen_args.items() -%} + - "--{{ key }}={{ value }}" + {% endfor %} + + hostNetwork: true + restartPolicy: OnFailure + ttlSecondsAfterFinished: 1800 diff --git a/kubespray/roles/network_plugin/cilium/templates/hubble/deploy.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/hubble/deploy.yml.j2 new file mode 100644 index 0000000..86533e6 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/hubble/deploy.yml.j2 @@ -0,0 +1,192 @@ +--- +# Source: cilium/templates/hubble-relay-deployment.yaml +apiVersion: apps/v1 +kind: Deployment +metadata: + name: hubble-relay + labels: + k8s-app: hubble-relay + namespace: kube-system +spec: + replicas: 1 + selector: + matchLabels: + k8s-app: hubble-relay + strategy: + rollingUpdate: + maxUnavailable: 1 + type: RollingUpdate + template: + metadata: + annotations: + labels: + k8s-app: hubble-relay + spec: + affinity: + podAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: "k8s-app" + operator: In + values: + - cilium + topologyKey: "kubernetes.io/hostname" + containers: + - name: hubble-relay + image: "{{ cilium_hubble_relay_image_repo }}:{{ cilium_hubble_relay_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + command: + - hubble-relay + args: + - serve + ports: + - name: grpc + containerPort: 4245 + readinessProbe: + tcpSocket: + port: grpc + livenessProbe: + tcpSocket: + port: grpc + volumeMounts: + - mountPath: /var/run/cilium + name: hubble-sock-dir + readOnly: true + - mountPath: /etc/hubble-relay + name: config + readOnly: true + {% if cilium_hubble_tls_generate -%} + - mountPath: /var/lib/hubble-relay/tls + name: tls + readOnly: true + {%- endif %} + + restartPolicy: Always + serviceAccount: hubble-relay + serviceAccountName: hubble-relay + terminationGracePeriodSeconds: 0 + volumes: + - configMap: + name: hubble-relay-config + items: + - key: config.yaml + path: config.yaml + name: config + - hostPath: + path: /var/run/cilium + type: Directory + name: hubble-sock-dir + {% if cilium_hubble_tls_generate -%} + - projected: + sources: + - secret: + name: hubble-relay-client-certs + items: + - key: ca.crt + path: hubble-server-ca.crt + - key: tls.crt + path: client.crt + - key: tls.key + path: client.key + - secret: + name: hubble-server-certs + items: + - key: tls.crt + path: server.crt + - key: tls.key + path: server.key + name: tls + {%- endif %} + +--- +# Source: cilium/templates/hubble-ui/deployment.yaml +kind: Deployment +apiVersion: apps/v1 +metadata: + namespace: kube-system + labels: + k8s-app: hubble-ui + name: hubble-ui +spec: + replicas: 1 + selector: + matchLabels: + k8s-app: hubble-ui + template: + metadata: + annotations: + labels: + k8s-app: hubble-ui + spec: + securityContext: + runAsUser: 1001 + serviceAccount: hubble-ui + serviceAccountName: hubble-ui + containers: + - name: frontend + image: "{{ cilium_hubble_ui_image_repo }}:{{ cilium_hubble_ui_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + ports: + - containerPort: 8081 + name: http + volumeMounts: + - name: hubble-ui-nginx-conf + mountPath: /etc/nginx/conf.d/default.conf + subPath: nginx.conf + - name: tmp-dir + mountPath: /tmp + resources: + {} + - name: backend + image: "{{ cilium_hubble_ui_backend_image_repo }}:{{ cilium_hubble_ui_backend_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + env: + - name: EVENTS_SERVER_PORT + value: "8090" + {% if cilium_hubble_tls_generate -%} + - name: TLS_TO_RELAY_ENABLED + value: "true" + - name: FLOWS_API_ADDR + value: "hubble-relay:443" + - name: TLS_RELAY_SERVER_NAME + value: ui.{{ cilium_cluster_name }}.hubble-grpc.cilium.io + - name: TLS_RELAY_CA_CERT_FILES + value: /var/lib/hubble-ui/certs/hubble-server-ca.crt + - name: TLS_RELAY_CLIENT_CERT_FILE + value: /var/lib/hubble-ui/certs/client.crt + - name: TLS_RELAY_CLIENT_KEY_FILE + value: /var/lib/hubble-ui/certs/client.key + {% else -%} + - name: FLOWS_API_ADDR + value: "hubble-relay:80" + {% endif %} + + volumeMounts: + - name: tls + mountPath: /var/lib/hubble-ui/certs + readOnly: true + ports: + - containerPort: 8090 + name: grpc + resources: + {} + volumes: + - configMap: + defaultMode: 420 + name: hubble-ui-nginx + name: hubble-ui-nginx-conf + - projected: + sources: + - secret: + name: hubble-relay-client-certs + items: + - key: ca.crt + path: hubble-server-ca.crt + - key: tls.crt + path: client.crt + - key: tls.key + path: client.key + name: tls + - emptyDir: {} + name: tmp-dir diff --git a/kubespray/roles/network_plugin/cilium/templates/hubble/job.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/hubble/job.yml.j2 new file mode 100644 index 0000000..9ad3ae3 --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/hubble/job.yml.j2 @@ -0,0 +1,34 @@ +--- +# Source: cilium/templates/hubble-generate-certs-job.yaml +apiVersion: batch/v1 +kind: Job +metadata: + name: hubble-generate-certs + namespace: kube-system + labels: + k8s-app: hubble-generate-certs +spec: + template: + metadata: + labels: + k8s-app: hubble-generate-certs + spec: + serviceAccount: hubble-generate-certs + serviceAccountName: hubble-generate-certs + containers: + - name: certgen + image: "{{ cilium_hubble_certgen_image_repo }}:{{ cilium_hubble_certgen_image_tag }}" + imagePullPolicy: {{ k8s_image_pull_policy }} + command: + - "/usr/bin/cilium-certgen" + # Because this is executed as a job, we pass the values as command + # line args instead of via config map. This allows users to inspect + # the values used in past runs by inspecting the completed pod. + args: + {% for key, value in cilium_certgen_args.items() -%} + - "--{{ key }}={{ value }}" + {% endfor %} + + hostNetwork: true + restartPolicy: OnFailure + ttlSecondsAfterFinished: 1800 diff --git a/kubespray/roles/network_plugin/cilium/templates/hubble/sa.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/hubble/sa.yml.j2 new file mode 100644 index 0000000..9b3203d --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/hubble/sa.yml.j2 @@ -0,0 +1,23 @@ +{% if cilium_hubble_tls_generate %} +--- +# Source: cilium/templates/hubble-generate-certs-serviceaccount.yaml +apiVersion: v1 +kind: ServiceAccount +metadata: + name: hubble-generate-certs + namespace: kube-system +{% endif %} +--- +# Source: cilium/templates/hubble-relay-serviceaccount.yaml +apiVersion: v1 +kind: ServiceAccount +metadata: + name: hubble-relay + namespace: kube-system +--- +# Source: cilium/templates/hubble-ui-serviceaccount.yaml +apiVersion: v1 +kind: ServiceAccount +metadata: + name: hubble-ui + namespace: kube-system diff --git a/kubespray/roles/network_plugin/cilium/templates/hubble/service.yml.j2 b/kubespray/roles/network_plugin/cilium/templates/hubble/service.yml.j2 new file mode 100644 index 0000000..f1df0eb --- /dev/null +++ b/kubespray/roles/network_plugin/cilium/templates/hubble/service.yml.j2 @@ -0,0 +1,102 @@ +{% if cilium_enable_prometheus or cilium_enable_hubble_metrics %} +--- +# Source: cilium/templates/cilium-agent-service.yaml +kind: Service +apiVersion: v1 +metadata: + name: hubble-metrics + namespace: kube-system + annotations: + prometheus.io/scrape: 'true' + prometheus.io/port: "9091" + labels: + k8s-app: hubble +spec: + clusterIP: None + type: ClusterIP + ports: + - name: hubble-metrics + port: 9091 + protocol: TCP + targetPort: hubble-metrics + selector: + k8s-app: cilium +--- +# Source: cilium/templates/hubble-relay/metrics-service.yaml +# We use a separate service from hubble-relay which can be exposed externally +kind: Service +apiVersion: v1 +metadata: + name: hubble-relay-metrics + namespace: kube-system + labels: + k8s-app: hubble-relay +spec: + clusterIP: None + type: ClusterIP + selector: + k8s-app: hubble-relay + ports: + - name: metrics + port: 9966 + protocol: TCP + targetPort: prometheus + +{% endif %} +--- +# Source: cilium/templates/hubble-relay-service.yaml +kind: Service +apiVersion: v1 +metadata: + name: hubble-relay + namespace: kube-system + labels: + k8s-app: hubble-relay +spec: + type: ClusterIP + selector: + k8s-app: hubble-relay + ports: + - protocol: TCP + {% if cilium_hubble_tls_generate -%} + port: 443 + {% else -%} + port: 80 + {% endif -%} + targetPort: 4245 +--- +# Source: cilium/templates/hubble-ui-service.yaml +kind: Service +apiVersion: v1 +metadata: + name: hubble-ui + labels: + k8s-app: hubble-ui + namespace: kube-system +spec: + selector: + k8s-app: hubble-ui + ports: + - name: http + port: 80 + targetPort: 8081 + type: ClusterIP +--- +# Source: cilium/templates/hubble/peer-service.yaml +apiVersion: v1 +kind: Service +metadata: + name: hubble-peer + namespace: kube-system + labels: + k8s-app: cilium +spec: + selector: + k8s-app: cilium + ports: + - name: peer-service + port: 443 + protocol: TCP + targetPort: 4244 + internalTrafficPolicy: Local + diff --git a/kubespray/roles/network_plugin/cni/tasks/main.yml b/kubespray/roles/network_plugin/cni/tasks/main.yml new file mode 100644 index 0000000..d74f169 --- /dev/null +++ b/kubespray/roles/network_plugin/cni/tasks/main.yml @@ -0,0 +1,16 @@ +--- +- name: CNI | make sure /opt/cni/bin exists + file: + path: /opt/cni/bin + state: directory + mode: 0755 + owner: "{{ kube_owner }}" + recurse: true + +- name: CNI | Copy cni plugins + unarchive: + src: "{{ downloads.cni.dest }}" + dest: "/opt/cni/bin" + mode: 0755 + owner: "{{ kube_owner }}" + remote_src: yes diff --git a/kubespray/roles/network_plugin/custom_cni/defaults/main.yml b/kubespray/roles/network_plugin/custom_cni/defaults/main.yml new file mode 100644 index 0000000..5cde372 --- /dev/null +++ b/kubespray/roles/network_plugin/custom_cni/defaults/main.yml @@ -0,0 +1,3 @@ +--- + +custom_cni_manifests: [] diff --git a/kubespray/roles/network_plugin/custom_cni/tasks/main.yml b/kubespray/roles/network_plugin/custom_cni/tasks/main.yml new file mode 100644 index 0000000..c428944 --- /dev/null +++ b/kubespray/roles/network_plugin/custom_cni/tasks/main.yml @@ -0,0 +1,26 @@ +--- +- name: Custom CNI | Check Custom CNI Manifests + assert: + that: + - "custom_cni_manifests | length > 0" + msg: "custom_cni_manifests should not be empty" + +- name: Custom CNI | Copy Custom manifests + template: + src: "{{ item }}" + dest: "{{ kube_config_dir }}/{{ item | basename | replace('.j2', '') }}" + mode: 0644 + loop: "{{ custom_cni_manifests }}" + delegate_to: "{{ groups['kube_control_plane'] | first }}" + run_once: true + +- name: Custom CNI | Start Resources + kube: + namespace: "kube-system" + kubectl: "{{ bin_dir }}/kubectl" + filename: "{{ kube_config_dir }}/{{ item | basename | replace('.j2', '') }}" + state: "latest" + wait: true + loop: "{{ custom_cni_manifests }}" + delegate_to: "{{ groups['kube_control_plane'] | first }}" + run_once: true diff --git a/kubespray/roles/network_plugin/flannel/defaults/main.yml b/kubespray/roles/network_plugin/flannel/defaults/main.yml new file mode 100644 index 0000000..8d7713b --- /dev/null +++ b/kubespray/roles/network_plugin/flannel/defaults/main.yml @@ -0,0 +1,28 @@ +--- +# Flannel public IP +# The address that flannel should advertise as how to access the system +# Disabled until https://github.com/coreos/flannel/issues/712 is fixed +# flannel_public_ip: "{{ access_ip | default(ip | default(fallback_ips[inventory_hostname])) }}" + +## interface that should be used for flannel operations +## This is actually an inventory cluster-level item +# flannel_interface: + +## Select interface that should be used for flannel operations by regexp on Name or IP +## This is actually an inventory cluster-level item +## example: select interface with ip from net 10.0.0.0/23 +## single quote and escape backslashes +# flannel_interface_regexp: '10\\.0\\.[0-2]\\.\\d{1,3}' + +# You can choose what type of flannel backend to use +# please refer to flannel's docs : https://github.com/coreos/flannel/blob/master/README.md +flannel_backend_type: "vxlan" +flannel_vxlan_vni: 1 +flannel_vxlan_port: 8472 +flannel_vxlan_direct_routing: false + +# Limits for apps +flannel_memory_limit: 500M +flannel_cpu_limit: 300m +flannel_memory_requests: 64M +flannel_cpu_requests: 150m diff --git a/kubespray/roles/network_plugin/flannel/meta/main.yml b/kubespray/roles/network_plugin/flannel/meta/main.yml new file mode 100644 index 0000000..9b7065f --- /dev/null +++ b/kubespray/roles/network_plugin/flannel/meta/main.yml @@ -0,0 +1,3 @@ +--- +dependencies: + - role: network_plugin/cni diff --git a/kubespray/roles/network_plugin/flannel/tasks/main.yml b/kubespray/roles/network_plugin/flannel/tasks/main.yml new file mode 100644 index 0000000..2fd82e9 --- /dev/null +++ b/kubespray/roles/network_plugin/flannel/tasks/main.yml @@ -0,0 +1,21 @@ +--- + +- name: Flannel | Stop if kernel version is too low for Flannel Wireguard encryption + assert: + that: ansible_kernel.split('-')[0] is version('5.6.0', '>=') + when: + - kube_network_plugin == 'flannel' + - flannel_backend_type == 'wireguard' + - not ignore_assert_errors + +- name: Flannel | Create Flannel manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: flannel, file: cni-flannel-rbac.yml, type: sa} + - {name: kube-flannel, file: cni-flannel.yml, type: ds} + register: flannel_node_manifests + when: + - inventory_hostname == groups['kube_control_plane'][0] diff --git a/kubespray/roles/network_plugin/flannel/tasks/reset.yml b/kubespray/roles/network_plugin/flannel/tasks/reset.yml new file mode 100644 index 0000000..03d40a0 --- /dev/null +++ b/kubespray/roles/network_plugin/flannel/tasks/reset.yml @@ -0,0 +1,24 @@ +--- +- name: Reset | check cni network device + stat: + path: /sys/class/net/cni0 + get_attributes: no + get_checksum: no + get_mime: no + register: cni + +- name: Reset | remove the network device created by the flannel + command: ip link del cni0 + when: cni.stat.exists + +- name: Reset | check flannel network device + stat: + path: /sys/class/net/flannel.1 + get_attributes: no + get_checksum: no + get_mime: no + register: flannel + +- name: Reset | remove the network device created by the flannel + command: ip link del flannel.1 + when: flannel.stat.exists diff --git a/kubespray/roles/network_plugin/flannel/templates/cni-flannel-rbac.yml.j2 b/kubespray/roles/network_plugin/flannel/templates/cni-flannel-rbac.yml.j2 new file mode 100644 index 0000000..631ec5e --- /dev/null +++ b/kubespray/roles/network_plugin/flannel/templates/cni-flannel-rbac.yml.j2 @@ -0,0 +1,52 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: flannel + namespace: kube-system +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: flannel +rules: +- apiGroups: + - "" + resources: + - pods + verbs: + - get +- apiGroups: + - "" + resources: + - nodes + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + - nodes/status + verbs: + - patch +- apiGroups: + - "networking.k8s.io" + resources: + - clustercidrs + verbs: + - list + - watch +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: flannel +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: flannel +subjects: +- kind: ServiceAccount + name: flannel + namespace: kube-system diff --git a/kubespray/roles/network_plugin/flannel/templates/cni-flannel.yml.j2 b/kubespray/roles/network_plugin/flannel/templates/cni-flannel.yml.j2 new file mode 100644 index 0000000..9c36d01 --- /dev/null +++ b/kubespray/roles/network_plugin/flannel/templates/cni-flannel.yml.j2 @@ -0,0 +1,170 @@ +--- +kind: ConfigMap +apiVersion: v1 +metadata: + name: kube-flannel-cfg + namespace: kube-system + labels: + tier: node + app: flannel +data: + cni-conf.json: | + { + "name": "cbr0", + "cniVersion": "0.3.1", + "plugins": [ + { + "type": "flannel", + "delegate": { + "hairpinMode": true, + "isDefaultGateway": true + } + }, + { + "type": "portmap", + "capabilities": { + "portMappings": true + } + } + ] + } + net-conf.json: | + { + "Network": "{{ kube_pods_subnet }}", + "EnableIPv4": true, +{% if enable_dual_stack_networks %} + "EnableIPv6": true, + "IPv6Network": "{{ kube_pods_subnet_ipv6 }}", +{% endif %} + "Backend": { + "Type": "{{ flannel_backend_type }}"{% if flannel_backend_type == "vxlan" %}, + "VNI": {{ flannel_vxlan_vni }}, + "Port": {{ flannel_vxlan_port }}, + "DirectRouting": {{ flannel_vxlan_direct_routing | to_json }} +{% endif %} + } + } +{% for arch in ['amd64', 'arm64', 'arm', 'ppc64le', 's390x'] %} +--- +apiVersion: apps/v1 +kind: DaemonSet +metadata: +{% if arch == 'amd64' %} + name: kube-flannel +{% else %} + name: kube-flannel-ds-{{ arch }} +{% endif %} + namespace: kube-system + labels: + tier: node + app: flannel +spec: + selector: + matchLabels: + app: flannel + template: + metadata: + labels: + tier: node + app: flannel + spec: + priorityClassName: system-node-critical + serviceAccountName: flannel + containers: + - name: kube-flannel + image: {{ flannel_image_repo }}:{{ flannel_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + resources: + limits: + cpu: {{ flannel_cpu_limit }} + memory: {{ flannel_memory_limit }} + requests: + cpu: {{ flannel_cpu_requests }} + memory: {{ flannel_memory_requests }} + command: [ "/opt/bin/flanneld", "--ip-masq", "--kube-subnet-mgr"{% if flannel_interface is defined %}, "--iface={{ flannel_interface }}"{% endif %}{% if flannel_interface_regexp is defined %}, "--iface-regex={{ flannel_interface_regexp }}"{% endif %} ] + securityContext: + privileged: false + capabilities: + add: ["NET_ADMIN", "NET_RAW"] + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + - name: EVENT_QUEUE_DEPTH + value: "5000" + volumeMounts: + - name: run + mountPath: /run/flannel + - name: flannel-cfg + mountPath: /etc/kube-flannel/ + - name: xtables-lock + mountPath: /run/xtables.lock + affinity: + nodeAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: kubernetes.io/os + operator: In + values: + - linux + - key: kubernetes.io/arch + operator: In + values: + - {{ arch }} + initContainers: + - name: install-cni-plugin + image: {{ flannel_init_image_repo }}:{{ flannel_init_image_tag }} + command: + - cp + args: + - -f + - /flannel + - /opt/cni/bin/flannel + volumeMounts: + - name: cni-plugin + mountPath: /opt/cni/bin + - name: install-cni + image: {{ flannel_image_repo }}:{{ flannel_image_tag }} + command: + - cp + args: + - -f + - /etc/kube-flannel/cni-conf.json + - /etc/cni/net.d/10-flannel.conflist + volumeMounts: + - name: cni + mountPath: /etc/cni/net.d + - name: flannel-cfg + mountPath: /etc/kube-flannel/ + hostNetwork: true + dnsPolicy: ClusterFirstWithHostNet + tolerations: + - operator: Exists + volumes: + - name: run + hostPath: + path: /run/flannel + - name: cni + hostPath: + path: /etc/cni/net.d + - name: flannel-cfg + configMap: + name: kube-flannel-cfg + - name: xtables-lock + hostPath: + path: /run/xtables.lock + type: FileOrCreate + - name: cni-plugin + hostPath: + path: /opt/cni/bin + updateStrategy: + rollingUpdate: + maxUnavailable: {{ serial | default('20%') }} + type: RollingUpdate +{% endfor %} diff --git a/kubespray/roles/network_plugin/kube-ovn/OWNERS b/kubespray/roles/network_plugin/kube-ovn/OWNERS new file mode 100644 index 0000000..84256aa --- /dev/null +++ b/kubespray/roles/network_plugin/kube-ovn/OWNERS @@ -0,0 +1,4 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +emeritus_approvers: +- oilbeater diff --git a/kubespray/roles/network_plugin/kube-ovn/defaults/main.yml b/kubespray/roles/network_plugin/kube-ovn/defaults/main.yml new file mode 100644 index 0000000..44850e5 --- /dev/null +++ b/kubespray/roles/network_plugin/kube-ovn/defaults/main.yml @@ -0,0 +1,118 @@ +--- +kube_ovn_db_cpu_request: 500m +kube_ovn_db_memory_request: 200Mi +kube_ovn_db_cpu_limit: 3000m +kube_ovn_db_memory_limit: 3000Mi +kube_ovn_node_cpu_request: 200m +kube_ovn_node_memory_request: 200Mi +kube_ovn_node_cpu_limit: 1000m +kube_ovn_node_memory_limit: 800Mi +kube_ovn_cni_server_cpu_request: 200m +kube_ovn_cni_server_memory_request: 200Mi +kube_ovn_cni_server_cpu_limit: 1000m +kube_ovn_cni_server_memory_limit: 1Gi +kube_ovn_controller_cpu_request: 200m +kube_ovn_controller_memory_request: 200Mi +kube_ovn_controller_cpu_limit: 1000m +kube_ovn_controller_memory_limit: 1Gi +kube_ovn_pinger_cpu_request: 100m +kube_ovn_pinger_memory_request: 200Mi +kube_ovn_pinger_cpu_limit: 200m +kube_ovn_pinger_memory_limit: 400Mi +kube_ovn_monitor_memory_request: 200Mi +kube_ovn_monitor_cpu_request: 200m +kube_ovn_monitor_memory_limit: 200Mi +kube_ovn_monitor_cpu_limit: 200m +kube_ovn_dpdk_node_cpu_request: 1000m +kube_ovn_dpdk_node_memory_request: 2Gi +kube_ovn_dpdk_node_cpu_limit: 1000m +kube_ovn_dpdk_node_memory_limit: 2Gi + +kube_ovn_central_hosts: "{{ groups['kube_control_plane'] }}" +kube_ovn_central_replics: "{{ kube_ovn_central_hosts | length }}" +kube_ovn_controller_replics: "{{ kube_ovn_central_hosts | length }}" +kube_ovn_central_ips: |- + {% for item in kube_ovn_central_hosts -%} + {{ hostvars[item]['ip'] | default(fallback_ips[item]) }}{% if not loop.last %},{% endif %} + {%- endfor %} + +kube_ovn_ic_enable: false +kube_ovn_ic_autoroute: true +kube_ovn_ic_dbhost: "127.0.0.1" +kube_ovn_ic_zone: "kubernetes" + +# geneve or vlan +kube_ovn_network_type: geneve + +# geneve, vxlan or stt. ATTENTION: some networkpolicy cannot take effect when using vxlan and stt need custom compile ovs kernel module +kube_ovn_tunnel_type: geneve + +## The nic to support container network can be a nic name or a group of regex separated by comma e.g: 'enp6s0f0,eth.*', if empty will use the nic that the default route use. +# kube_ovn_iface: eth1 +## The MTU used by pod iface in overlay networks (default iface MTU - 100) +# kube_ovn_mtu: 1333 + +## Enable hw-offload, disable traffic mirror and set the iface to the physical port. Make sure that there is an IP address bind to the physical port. +kube_ovn_hw_offload: false +# traffic mirror +kube_ovn_traffic_mirror: false + +# kube_ovn_pool_cidr_ipv6: fd85:ee78:d8a6:8607::1:0000/112 +# kube_ovn_default_interface_name: eth0 + +kube_ovn_external_address: 8.8.8.8 +kube_ovn_external_address_ipv6: 2400:3200::1 +kube_ovn_external_dns: alauda.cn + +# kube_ovn_default_gateway: 10.233.64.1,fd85:ee78:d8a6:8607::1:0 +kube_ovn_default_gateway_check: true +kube_ovn_default_logical_gateway: false + +# u2o_interconnection +kube_ovn_u2o_interconnection: false + +# kube_ovn_default_exclude_ips: 10.16.0.1 +kube_ovn_node_switch_cidr: 100.64.0.0/16 +kube_ovn_node_switch_cidr_ipv6: fd00:100:64::/64 + +## vlan config, set default interface name and vlan id +# kube_ovn_default_interface_name: eth0 +kube_ovn_default_vlan_id: 100 +kube_ovn_vlan_name: product + +## pod nic type, support: veth-pair or internal-port +kube_ovn_pod_nic_type: veth_pair + +## Enable load balancer +kube_ovn_enable_lb: true + +## Enable network policy support +kube_ovn_enable_np: true + +## Enable external vpc support +kube_ovn_enable_external_vpc: true + +## Enable checksum +kube_ovn_encap_checksum: true + +## enable ssl +kube_ovn_enable_ssl: false + +## dpdk +kube_ovn_dpdk_enabled: false +kube_ovn_dpdk_tunnel_iface: br-phy + +## bind local ip +kube_ovn_bind_local_ip_enabled: true + +## eip snat +kube_ovn_eip_snat_enabled: true + +# ls dnat mod dl dst +kube_ovn_ls_dnat_mod_dl_dst: true + +## keep vm ip +kube_ovn_keep_vm_ip: true + +## cni config priority, default: 01 +kube_ovn_cni_config_priority: 01 diff --git a/kubespray/roles/network_plugin/kube-ovn/tasks/main.yml b/kubespray/roles/network_plugin/kube-ovn/tasks/main.yml new file mode 100644 index 0000000..ab45b62 --- /dev/null +++ b/kubespray/roles/network_plugin/kube-ovn/tasks/main.yml @@ -0,0 +1,17 @@ +--- +- name: Kube-OVN | Label ovn-db node + command: "{{ kubectl }} label --overwrite node {{ item }} kube-ovn/role=master" + loop: "{{ kube_ovn_central_hosts }}" + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Kube-OVN | Create Kube-OVN manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: kube-ovn-crd, file: cni-kube-ovn-crd.yml} + - {name: ovn, file: cni-ovn.yml} + - {name: kube-ovn, file: cni-kube-ovn.yml} + register: kube_ovn_node_manifests diff --git a/kubespray/roles/network_plugin/kube-ovn/templates/cni-kube-ovn-crd.yml.j2 b/kubespray/roles/network_plugin/kube-ovn/templates/cni-kube-ovn-crd.yml.j2 new file mode 100644 index 0000000..379381d --- /dev/null +++ b/kubespray/roles/network_plugin/kube-ovn/templates/cni-kube-ovn-crd.yml.j2 @@ -0,0 +1,1533 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: vpc-dnses.kubeovn.io +spec: + group: kubeovn.io + names: + plural: vpc-dnses + singular: vpc-dns + shortNames: + - vpc-dns + kind: VpcDns + listKind: VpcDnsList + scope: Cluster + versions: + - additionalPrinterColumns: + - jsonPath: .status.active + name: Active + type: boolean + - jsonPath: .spec.vpc + name: Vpc + type: string + - jsonPath: .spec.subnet + name: Subnet + type: string + name: v1 + served: true + storage: true + subresources: + status: {} + schema: + openAPIV3Schema: + type: object + properties: + spec: + type: object + properties: + vpc: + type: string + subnet: + type: string + status: + type: object + properties: + active: + type: boolean + conditions: + type: array + items: + type: object + properties: + type: + type: string + status: + type: string + reason: + type: string + message: + type: string + lastUpdateTime: + type: string + lastTransitionTime: + type: string +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: switch-lb-rules.kubeovn.io +spec: + group: kubeovn.io + names: + plural: switch-lb-rules + singular: switch-lb-rule + shortNames: + - slr + kind: SwitchLBRule + listKind: SwitchLBRuleList + scope: Cluster + versions: + - additionalPrinterColumns: + - jsonPath: .spec.vip + name: vip + type: string + - jsonPath: .status.ports + name: port(s) + type: string + - jsonPath: .status.service + name: service + type: string + - jsonPath: .metadata.creationTimestamp + name: age + type: date + name: v1 + served: true + storage: true + subresources: + status: {} + schema: + openAPIV3Schema: + type: object + properties: + spec: + type: object + properties: + namespace: + type: string + vip: + type: string + sessionAffinity: + type: string + ports: + items: + properties: + name: + type: string + port: + type: integer + minimum: 1 + maximum: 65535 + protocol: + type: string + targetPort: + type: integer + minimum: 1 + maximum: 65535 + type: object + type: array + selector: + items: + type: string + type: array + status: + type: object + properties: + ports: + type: string + service: + type: string +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: vpc-nat-gateways.kubeovn.io +spec: + group: kubeovn.io + names: + plural: vpc-nat-gateways + singular: vpc-nat-gateway + shortNames: + - vpc-nat-gw + kind: VpcNatGateway + listKind: VpcNatGatewayList + scope: Cluster + versions: + - additionalPrinterColumns: + - jsonPath: .spec.vpc + name: Vpc + type: string + - jsonPath: .spec.subnet + name: Subnet + type: string + - jsonPath: .spec.lanIp + name: LanIP + type: string + name: v1 + served: true + storage: true + schema: + openAPIV3Schema: + type: object + properties: + spec: + type: object + properties: + lanIp: + type: string + subnet: + type: string + vpc: + type: string + selector: + type: array + items: + type: string + tolerations: + type: array + items: + type: object + properties: + key: + type: string + operator: + type: string + value: + type: string + effect: + type: string + tolerationSeconds: + type: integer +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: iptables-eips.kubeovn.io +spec: + group: kubeovn.io + names: + plural: iptables-eips + singular: iptables-eip + shortNames: + - eip + kind: IptablesEIP + listKind: IptablesEIPList + scope: Cluster + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - jsonPath: .status.ip + name: IP + type: string + - jsonPath: .spec.macAddress + name: Mac + type: string + - jsonPath: .status.nat + name: Nat + type: string + - jsonPath: .spec.natGwDp + name: NatGwDp + type: string + - jsonPath: .status.ready + name: Ready + type: boolean + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + ready: + type: boolean + ip: + type: string + nat: + type: string + redo: + type: string + conditions: + type: array + items: + type: object + properties: + type: + type: string + status: + type: string + reason: + type: string + message: + type: string + lastUpdateTime: + type: string + lastTransitionTime: + type: string + spec: + type: object + properties: + v4ip: + type: string + v6ip: + type: string + macAddress: + type: string + natGwDp: + type: string +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: iptables-fip-rules.kubeovn.io +spec: + group: kubeovn.io + names: + plural: iptables-fip-rules + singular: iptables-fip-rule + shortNames: + - fip + kind: IptablesFIPRule + listKind: IptablesFIPRuleList + scope: Cluster + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - jsonPath: .spec.eip + name: Eip + type: string + - jsonPath: .status.v4ip + name: V4ip + type: string + - jsonPath: .spec.internalIp + name: InternalIp + type: string + - jsonPath: .status.v6ip + name: V6ip + type: string + - jsonPath: .status.ready + name: Ready + type: boolean + - jsonPath: .status.natGwDp + name: NatGwDp + type: string + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + ready: + type: boolean + v4ip: + type: string + v6ip: + type: string + natGwDp: + type: string + redo: + type: string + conditions: + type: array + items: + type: object + properties: + type: + type: string + status: + type: string + reason: + type: string + message: + type: string + lastUpdateTime: + type: string + lastTransitionTime: + type: string + spec: + type: object + properties: + eip: + type: string + internalIp: + type: string +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: iptables-dnat-rules.kubeovn.io +spec: + group: kubeovn.io + names: + plural: iptables-dnat-rules + singular: iptables-dnat-rule + shortNames: + - dnat + kind: IptablesDnatRule + listKind: IptablesDnatRuleList + scope: Cluster + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - jsonPath: .spec.eip + name: Eip + type: string + - jsonPath: .spec.protocol + name: Protocol + type: string + - jsonPath: .status.v4ip + name: V4ip + type: string + - jsonPath: .status.v6ip + name: V6ip + type: string + - jsonPath: .spec.internalIp + name: InternalIp + type: string + - jsonPath: .spec.externalPort + name: ExternalPort + type: string + - jsonPath: .spec.internalPort + name: InternalPort + type: string + - jsonPath: .status.natGwDp + name: NatGwDp + type: string + - jsonPath: .status.ready + name: Ready + type: boolean + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + ready: + type: boolean + v4ip: + type: string + v6ip: + type: string + natGwDp: + type: string + redo: + type: string + protocol: + type: string + internalIp: + type: string + internalPort: + type: string + externalPort: + type: string + conditions: + type: array + items: + type: object + properties: + type: + type: string + status: + type: string + reason: + type: string + message: + type: string + lastUpdateTime: + type: string + lastTransitionTime: + type: string + spec: + type: object + properties: + eip: + type: string + externalPort: + type: string + protocol: + type: string + internalIp: + type: string + internalPort: + type: string +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: iptables-snat-rules.kubeovn.io +spec: + group: kubeovn.io + names: + plural: iptables-snat-rules + singular: iptables-snat-rule + shortNames: + - snat + kind: IptablesSnatRule + listKind: IptablesSnatRuleList + scope: Cluster + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - jsonPath: .spec.eip + name: EIP + type: string + - jsonPath: .status.v4ip + name: V4ip + type: string + - jsonPath: .status.v6ip + name: V6ip + type: string + - jsonPath: .spec.internalCIDR + name: InternalCIDR + type: string + - jsonPath: .status.natGwDp + name: NatGwDp + type: string + - jsonPath: .status.ready + name: Ready + type: boolean + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + ready: + type: boolean + v4ip: + type: string + v6ip: + type: string + natGwDp: + type: string + redo: + type: string + conditions: + type: array + items: + type: object + properties: + type: + type: string + status: + type: string + reason: + type: string + message: + type: string + lastUpdateTime: + type: string + lastTransitionTime: + type: string + spec: + type: object + properties: + eip: + type: string + internalCIDR: + type: string +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: ovn-eips.kubeovn.io +spec: + group: kubeovn.io + names: + plural: ovn-eips + singular: ovn-eip + shortNames: + - oeip + kind: OvnEip + listKind: OvnEipList + scope: Cluster + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - jsonPath: .spec.v4ip + name: IP + type: string + - jsonPath: .spec.macAddress + name: Mac + type: string + - jsonPath: .spec.type + name: Type + type: string + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + v4Ip: + type: string + macAddress: + type: string + conditions: + type: array + items: + type: object + properties: + type: + type: string + status: + type: string + reason: + type: string + message: + type: string + lastUpdateTime: + type: string + lastTransitionTime: + type: string + spec: + type: object + properties: + externalSubnet: + type: string + type: + type: string + v4ip: + type: string + macAddress: + type: string +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: ovn-fips.kubeovn.io +spec: + group: kubeovn.io + names: + plural: ovn-fips + singular: ovn-fip + shortNames: + - ofip + kind: OvnFip + listKind: OvnFipList + scope: Cluster + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - jsonPath: .status.vpc + name: Vpc + type: string + - jsonPath: .status.v4Eip + name: V4Eip + type: string + - jsonPath: .status.v4Ip + name: V4Ip + type: string + - jsonPath: .status.ready + name: Ready + type: boolean + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + ready: + type: boolean + v4Eip: + type: string + v4Ip: + type: string + macAddress: + type: string + vpc: + type: string + conditions: + type: array + items: + type: object + properties: + type: + type: string + status: + type: string + reason: + type: string + message: + type: string + lastUpdateTime: + type: string + lastTransitionTime: + type: string + spec: + type: object + properties: + ovnEip: + type: string + ipName: + type: string +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: ovn-snat-rules.kubeovn.io +spec: + group: kubeovn.io + names: + plural: ovn-snat-rules + singular: ovn-snat-rule + shortNames: + - osnat + kind: OvnSnatRule + listKind: OvnSnatRuleList + scope: Cluster + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - jsonPath: .status.vpc + name: Vpc + type: string + - jsonPath: .status.v4Eip + name: V4Eip + type: string + - jsonPath: .status.v4ipCidr + name: V4Ip + type: string + - jsonPath: .status.ready + name: Ready + type: boolean + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + ready: + type: boolean + v4Eip: + type: string + v4ipCidr: + type: string + vpc: + type: string + conditions: + type: array + items: + type: object + properties: + type: + type: string + status: + type: string + reason: + type: string + message: + type: string + lastUpdateTime: + type: string + lastTransitionTime: + type: string + spec: + type: object + properties: + ovnEip: + type: string + vpcSubnet: + type: string + ipName: + type: string +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: vpcs.kubeovn.io +spec: + group: kubeovn.io + versions: + - additionalPrinterColumns: + - jsonPath: .status.enableExternal + name: EnableExternal + type: boolean + - jsonPath: .status.standby + name: Standby + type: boolean + - jsonPath: .status.subnets + name: Subnets + type: string + - jsonPath: .spec.namespaces + name: Namespaces + type: string + name: v1 + schema: + openAPIV3Schema: + properties: + spec: + properties: + enableExternal: + type: boolean + namespaces: + items: + type: string + type: array + staticRoutes: + items: + properties: + policy: + type: string + cidr: + type: string + nextHopIP: + type: string + type: object + type: array + policyRoutes: + items: + properties: + priority: + type: integer + action: + type: string + match: + type: string + nextHopIP: + type: string + type: object + type: array + vpcPeerings: + items: + properties: + remoteVpc: + type: string + localConnectIP: + type: string + type: object + type: array + type: object + status: + properties: + conditions: + items: + properties: + lastTransitionTime: + type: string + lastUpdateTime: + type: string + message: + type: string + reason: + type: string + status: + type: string + type: + type: string + type: object + type: array + default: + type: boolean + defaultLogicalSwitch: + type: string + router: + type: string + standby: + type: boolean + enableExternal: + type: boolean + subnets: + items: + type: string + type: array + vpcPeerings: + items: + type: string + type: array + tcpLoadBalancer: + type: string + tcpSessionLoadBalancer: + type: string + udpLoadBalancer: + type: string + udpSessionLoadBalancer: + type: string + sctpLoadBalancer: + type: string + sctpSessionLoadBalancer: + type: string + type: object + type: object + served: true + storage: true + subresources: + status: {} + names: + kind: Vpc + listKind: VpcList + plural: vpcs + shortNames: + - vpc + singular: vpc + scope: Cluster +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: ips.kubeovn.io +spec: + group: kubeovn.io + versions: + - name: v1 + served: true + storage: true + additionalPrinterColumns: + - name: V4IP + type: string + jsonPath: .spec.v4IpAddress + - name: V6IP + type: string + jsonPath: .spec.v6IpAddress + - name: Mac + type: string + jsonPath: .spec.macAddress + - name: Node + type: string + jsonPath: .spec.nodeName + - name: Subnet + type: string + jsonPath: .spec.subnet + schema: + openAPIV3Schema: + type: object + properties: + spec: + type: object + properties: + podName: + type: string + namespace: + type: string + subnet: + type: string + attachSubnets: + type: array + items: + type: string + nodeName: + type: string + ipAddress: + type: string + v4IpAddress: + type: string + v6IpAddress: + type: string + attachIps: + type: array + items: + type: string + macAddress: + type: string + attachMacs: + type: array + items: + type: string + containerID: + type: string + podType: + type: string + scope: Cluster + names: + plural: ips + singular: ip + kind: IP + shortNames: + - ip +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: vips.kubeovn.io +spec: + group: kubeovn.io + names: + plural: vips + singular: vip + shortNames: + - vip + kind: Vip + listKind: VipList + scope: Cluster + versions: + - name: v1 + served: true + storage: true + additionalPrinterColumns: + - name: V4IP + type: string + jsonPath: .status.v4ip + - name: PV4IP + type: string + jsonPath: .spec.parentV4ip + - name: Mac + type: string + jsonPath: .status.mac + - name: PMac + type: string + jsonPath: .spec.parentMac + - name: V6IP + type: string + jsonPath: .status.v6ip + - name: PV6IP + type: string + jsonPath: .spec.parentV6ip + - name: Subnet + type: string + jsonPath: .spec.subnet + - jsonPath: .status.ready + name: Ready + type: boolean + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + ready: + type: boolean + v4ip: + type: string + v6ip: + type: string + mac: + type: string + pv4ip: + type: string + pv6ip: + type: string + pmac: + type: string + conditions: + type: array + items: + type: object + properties: + type: + type: string + status: + type: string + reason: + type: string + message: + type: string + lastUpdateTime: + type: string + lastTransitionTime: + type: string + spec: + type: object + properties: + namespace: + type: string + subnet: + type: string + attachSubnets: + type: array + items: + type: string + v4ip: + type: string + macAddress: + type: string + v6ip: + type: string + parentV4ip: + type: string + parentMac: + type: string + parentV6ip: + type: string +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: subnets.kubeovn.io +spec: + group: kubeovn.io + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Provider + type: string + jsonPath: .spec.provider + - name: Vpc + type: string + jsonPath: .spec.vpc + - name: Protocol + type: string + jsonPath: .spec.protocol + - name: CIDR + type: string + jsonPath: .spec.cidrBlock + - name: Private + type: boolean + jsonPath: .spec.private + - name: NAT + type: boolean + jsonPath: .spec.natOutgoing + - name: Default + type: boolean + jsonPath: .spec.default + - name: GatewayType + type: string + jsonPath: .spec.gatewayType + - name: V4Used + type: number + jsonPath: .status.v4usingIPs + - name: V4Available + type: number + jsonPath: .status.v4availableIPs + - name: V6Used + type: number + jsonPath: .status.v6usingIPs + - name: V6Available + type: number + jsonPath: .status.v6availableIPs + - name: ExcludeIPs + type: string + jsonPath: .spec.excludeIps + - name: U2OInterconnectionIP + type: string + jsonPath: .status.u2oInterconnectionIP + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + v4availableIPs: + type: number + v4usingIPs: + type: number + v6availableIPs: + type: number + v6usingIPs: + type: number + activateGateway: + type: string + dhcpV4OptionsUUID: + type: string + dhcpV6OptionsUUID: + type: string + u2oInterconnectionIP: + type: string + conditions: + type: array + items: + type: object + properties: + type: + type: string + status: + type: string + reason: + type: string + message: + type: string + lastUpdateTime: + type: string + lastTransitionTime: + type: string + spec: + type: object + properties: + vpc: + type: string + default: + type: boolean + protocol: + type: string + enum: + - IPv4 + - IPv6 + - Dual + cidrBlock: + type: string + namespaces: + type: array + items: + type: string + gateway: + type: string + provider: + type: string + excludeIps: + type: array + items: + type: string + vips: + type: array + items: + type: string + gatewayType: + type: string + allowSubnets: + type: array + items: + type: string + gatewayNode: + type: string + natOutgoing: + type: boolean + u2oRouting: + type: boolean + externalEgressGateway: + type: string + policyRoutingPriority: + type: integer + minimum: 1 + maximum: 32765 + policyRoutingTableID: + type: integer + minimum: 1 + maximum: 2147483647 + not: + enum: + - 252 # compat + - 253 # default + - 254 # main + - 255 # local + private: + type: boolean + vlan: + type: string + logicalGateway: + type: boolean + disableGatewayCheck: + type: boolean + disableInterConnection: + type: boolean + enableDHCP: + type: boolean + dhcpV4Options: + type: string + dhcpV6Options: + type: string + enableIPv6RA: + type: boolean + ipv6RAConfigs: + type: string + acls: + type: array + items: + type: object + properties: + direction: + type: string + enum: + - from-lport + - to-lport + priority: + type: integer + minimum: 0 + maximum: 32767 + match: + type: string + action: + type: string + enum: + - allow-related + - allow-stateless + - allow + - drop + - reject + u2oInterconnection: + type: boolean + scope: Cluster + names: + plural: subnets + singular: subnet + kind: Subnet + shortNames: + - subnet +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: vlans.kubeovn.io +spec: + group: kubeovn.io + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + schema: + openAPIV3Schema: + type: object + properties: + spec: + type: object + properties: + id: + type: integer + minimum: 0 + maximum: 4095 + provider: + type: string + vlanId: + type: integer + description: Deprecated in favor of id + providerInterfaceName: + type: string + description: Deprecated in favor of provider + required: + - provider + status: + type: object + properties: + subnets: + type: array + items: + type: string + additionalPrinterColumns: + - name: ID + type: string + jsonPath: .spec.id + - name: Provider + type: string + jsonPath: .spec.provider + scope: Cluster + names: + plural: vlans + singular: vlan + kind: Vlan + shortNames: + - vlan +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: provider-networks.kubeovn.io +spec: + group: kubeovn.io + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + schema: + openAPIV3Schema: + type: object + properties: + metadata: + type: object + properties: + name: + type: string + maxLength: 12 + not: + enum: + - int + - external + spec: + type: object + properties: + defaultInterface: + type: string + maxLength: 15 + pattern: '^[^/\s]+$' + customInterfaces: + type: array + items: + type: object + properties: + interface: + type: string + maxLength: 15 + pattern: '^[^/\s]+$' + nodes: + type: array + items: + type: string + exchangeLinkName: + type: boolean + excludeNodes: + type: array + items: + type: string + required: + - defaultInterface + status: + type: object + properties: + ready: + type: boolean + readyNodes: + type: array + items: + type: string + notReadyNodes: + type: array + items: + type: string + vlans: + type: array + items: + type: string + conditions: + type: array + items: + type: object + properties: + node: + type: string + type: + type: string + status: + type: string + reason: + type: string + message: + type: string + lastUpdateTime: + type: string + lastTransitionTime: + type: string + additionalPrinterColumns: + - name: DefaultInterface + type: string + jsonPath: .spec.defaultInterface + - name: Ready + type: boolean + jsonPath: .status.ready + scope: Cluster + names: + plural: provider-networks + singular: provider-network + kind: ProviderNetwork + listKind: ProviderNetworkList +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: security-groups.kubeovn.io +spec: + group: kubeovn.io + names: + plural: security-groups + singular: security-group + shortNames: + - sg + kind: SecurityGroup + listKind: SecurityGroupList + scope: Cluster + versions: + - name: v1 + served: true + storage: true + schema: + openAPIV3Schema: + type: object + properties: + spec: + type: object + properties: + ingressRules: + type: array + items: + type: object + properties: + ipVersion: + type: string + protocol: + type: string + priority: + type: integer + remoteType: + type: string + remoteAddress: + type: string + remoteSecurityGroup: + type: string + portRangeMin: + type: integer + portRangeMax: + type: integer + policy: + type: string + egressRules: + type: array + items: + type: object + properties: + ipVersion: + type: string + protocol: + type: string + priority: + type: integer + remoteType: + type: string + remoteAddress: + type: string + remoteSecurityGroup: + type: string + portRangeMin: + type: integer + portRangeMax: + type: integer + policy: + type: string + allowSameGroupTraffic: + type: boolean + status: + type: object + properties: + portGroup: + type: string + allowSameGroupTraffic: + type: boolean + ingressMd5: + type: string + egressMd5: + type: string + ingressLastSyncSuccess: + type: boolean + egressLastSyncSuccess: + type: boolean + subresources: + status: {} + conversion: + strategy: None \ No newline at end of file diff --git a/kubespray/roles/network_plugin/kube-ovn/templates/cni-kube-ovn.yml.j2 b/kubespray/roles/network_plugin/kube-ovn/templates/cni-kube-ovn.yml.j2 new file mode 100644 index 0000000..cee7ccb --- /dev/null +++ b/kubespray/roles/network_plugin/kube-ovn/templates/cni-kube-ovn.yml.j2 @@ -0,0 +1,673 @@ +--- +kind: Deployment +apiVersion: apps/v1 +metadata: + name: kube-ovn-controller + namespace: kube-system + annotations: + kubernetes.io/description: | + kube-ovn controller +spec: + replicas: {{ kube_ovn_controller_replics }} + selector: + matchLabels: + app: kube-ovn-controller + strategy: + rollingUpdate: + maxSurge: 0% + maxUnavailable: 100% + type: RollingUpdate + template: + metadata: + labels: + app: kube-ovn-controller + component: network + type: infra + spec: + tolerations: + - operator: Exists + affinity: + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchLabels: + app: kube-ovn-controller + topologyKey: kubernetes.io/hostname + priorityClassName: system-cluster-critical + serviceAccountName: ovn + hostNetwork: true + containers: + - name: kube-ovn-controller + image: {{ kube_ovn_container_image_repo }}:{{ kube_ovn_container_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - /kube-ovn/start-controller.sh + - --default-cidr={{ kube_pods_subnet }}{% if enable_dual_stack_networks %},{{ kube_ovn_pool_cidr_ipv6 | default(kube_pods_subnet_ipv6) }}{% endif %}{{ '' }} + - --default-gateway={% if kube_ovn_default_gateway is defined %}{{ kube_ovn_default_gateway }}{% endif %}{{ '' }} + - --default-gateway-check={{ kube_ovn_default_gateway_check | string }} + - --default-logical-gateway={{ kube_ovn_default_logical_gateway | string }} + - --default-u2o-interconnection={{ kube_ovn_u2o_interconnection }} + - --default-exclude-ips={% if kube_ovn_default_exclude_ips is defined %}{{ kube_ovn_default_exclude_ips }}{% endif %}{{ '' }} + - --node-switch-cidr={{ kube_ovn_node_switch_cidr }}{% if enable_dual_stack_networks %},{{ kube_ovn_node_switch_cidr_ipv6 }}{% endif %}{{ '' }} + - --service-cluster-ip-range={{ kube_service_addresses }}{% if enable_dual_stack_networks %},{{ kube_service_addresses_ipv6 }}{% endif %}{{ '' }} + - --network-type={{ kube_ovn_network_type }} + - --default-interface-name={{ kube_ovn_default_interface_name | default('') }} + - --default-vlan-id={{ kube_ovn_default_vlan_id }} + - --ls-dnat-mod-dl-dst={{ kube_ovn_ls_dnat_mod_dl_dst }} + - --pod-nic-type={{ kube_ovn_pod_nic_type }} + - --enable-lb={{ kube_ovn_enable_lb | string }} + - --enable-np={{ kube_ovn_enable_np | string }} + - --enable-eip-snat={{ kube_ovn_eip_snat_enabled }} + - --enable-external-vpc={{ kube_ovn_enable_external_vpc | string }} + - --logtostderr=false + - --alsologtostderr=true + - --gc-interval=360 + - --inspect-interval=20 + - --log_file=/var/log/kube-ovn/kube-ovn-controller.log + - --log_file_max_size=0 + - --enable-lb-svc=false + - --keep-vm-ip={{ kube_ovn_keep_vm_ip }} + - --pod-default-fip-type="" + env: + - name: ENABLE_SSL + value: "{{ kube_ovn_enable_ssl | lower }}" + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: KUBE_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + - name: KUBE_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: OVN_DB_IPS + value: "{{ kube_ovn_central_ips }}" + - name: POD_IPS + valueFrom: + fieldRef: + fieldPath: status.podIPs + - name: ENABLE_BIND_LOCAL_IP + value: "{{ kube_ovn_bind_local_ip_enabled }}" + volumeMounts: + - mountPath: /etc/localtime + name: localtime + - mountPath: /var/log/kube-ovn + name: kube-ovn-log + - mountPath: /var/run/tls + name: kube-ovn-tls + readinessProbe: + exec: + command: + - /kube-ovn/kube-ovn-controller-healthcheck + periodSeconds: 3 + timeoutSeconds: 45 + livenessProbe: + exec: + command: + - /kube-ovn/kube-ovn-controller-healthcheck + initialDelaySeconds: 300 + periodSeconds: 7 + failureThreshold: 5 + timeoutSeconds: 45 + resources: + requests: + cpu: {{ kube_ovn_controller_cpu_request }} + memory: {{ kube_ovn_controller_memory_request }} + limits: + cpu: {{ kube_ovn_controller_cpu_limit }} + memory: {{ kube_ovn_controller_memory_limit }} + nodeSelector: + kubernetes.io/os: "linux" + volumes: + - name: localtime + hostPath: + path: /etc/localtime + - name: kube-ovn-log + hostPath: + path: /var/log/kube-ovn + - name: kube-ovn-tls + secret: + optional: true + secretName: kube-ovn-tls + +--- +kind: DaemonSet +apiVersion: apps/v1 +metadata: + name: kube-ovn-cni + namespace: kube-system + annotations: + kubernetes.io/description: | + This daemon set launches the kube-ovn cni daemon. +spec: + selector: + matchLabels: + app: kube-ovn-cni + template: + metadata: + labels: + app: kube-ovn-cni + component: network + type: infra + spec: + tolerations: + - effect: NoSchedule + operator: Exists + - effect: NoExecute + operator: Exists + - key: CriticalAddonsOnly + operator: Exists + priorityClassName: system-node-critical + serviceAccountName: ovn + hostNetwork: true + hostPID: true + initContainers: + - name: install-cni + image: {{ kube_ovn_container_image_repo }}:{{ kube_ovn_container_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + command: ["/kube-ovn/install-cni.sh"] + securityContext: + runAsUser: 0 + privileged: true + volumeMounts: + - mountPath: /opt/cni/bin + name: cni-bin + - mountPath: /usr/local/bin + name: local-bin + containers: + - name: cni-server + image: {{ kube_ovn_container_image_repo }}:{{ kube_ovn_container_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + command: + - bash + - /kube-ovn/start-cniserver.sh + args: + - --enable-mirror={{ kube_ovn_traffic_mirror | lower }} + - --encap-checksum={{ kube_ovn_encap_checksum | lower }} + - --service-cluster-ip-range={{ kube_service_addresses }}{% if enable_dual_stack_networks %},{{ kube_service_addresses_ipv6 }}{% endif %}{{ '' }} + - --iface={{ kube_ovn_iface | default('') }} + - --dpdk-tunnel-iface={{ kube_ovn_dpdk_tunnel_iface }} + - --network-type={{ kube_ovn_network_type }} + - --default-interface-name={{ kube_ovn_default_interface_name | default('') }} + {% if kube_ovn_mtu is defined %} + - --mtu={{ kube_ovn_mtu }} +{% endif %} + - --cni-conf-name={{ kube_ovn_cni_config_priority }}-kube-ovn.conflist + - --logtostderr=false + - --alsologtostderr=true + - --log_file=/var/log/kube-ovn/kube-ovn-cni.log + - --log_file_max_size=0 + securityContext: + runAsUser: 0 + privileged: true + env: + - name: ENABLE_SSL + value: "{{ kube_ovn_enable_ssl | lower }}" + - name: POD_IP + valueFrom: + fieldRef: + fieldPath: status.podIP + - name: KUBE_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: MODULES + value: kube_ovn_fastpath.ko + - name: RPMS + value: openvswitch-kmod + - name: POD_IPS + valueFrom: + fieldRef: + fieldPath: status.podIPs + - name: ENABLE_BIND_LOCAL_IP + value: "{{ kube_ovn_bind_local_ip_enabled }}" + - name: DBUS_SYSTEM_BUS_ADDRESS + value: "unix:path=/host/var/run/dbus/system_bus_socket" + volumeMounts: + - name: host-modules + mountPath: /lib/modules + readOnly: true + - name: shared-dir + mountPath: /var/lib/kubelet/pods + - mountPath: /etc/openvswitch + name: systemid + - mountPath: /etc/cni/net.d + name: cni-conf + - mountPath: /run/openvswitch + name: host-run-ovs + mountPropagation: Bidirectional + - mountPath: /run/ovn + name: host-run-ovn + - mountPath: /host/var/run/dbus + name: host-dbus + mountPropagation: HostToContainer + - mountPath: /var/run/netns + name: host-ns + mountPropagation: HostToContainer + - mountPath: /var/log/kube-ovn + name: kube-ovn-log + - mountPath: /var/log/openvswitch + name: host-log-ovs + - mountPath: /var/log/ovn + name: host-log-ovn + - mountPath: /etc/localtime + name: localtime + - mountPath: /tmp + name: tmp + livenessProbe: + failureThreshold: 3 + initialDelaySeconds: 30 + periodSeconds: 7 + successThreshold: 1 + tcpSocket: + port: 10665 + timeoutSeconds: 3 + readinessProbe: + failureThreshold: 3 + initialDelaySeconds: 30 + periodSeconds: 7 + successThreshold: 1 + tcpSocket: + port: 10665 + timeoutSeconds: 3 + resources: + requests: + cpu: {{ kube_ovn_cni_server_cpu_request }} + memory: {{ kube_ovn_cni_server_memory_request }} + limits: + cpu: {{ kube_ovn_cni_server_cpu_limit }} + memory: {{ kube_ovn_cni_server_memory_limit }} + nodeSelector: + kubernetes.io/os: "linux" + volumes: + - name: host-modules + hostPath: + path: /lib/modules + - name: shared-dir + hostPath: + path: /var/lib/kubelet/pods + - name: systemid + hostPath: + path: /etc/origin/openvswitch + - name: host-run-ovs + hostPath: + path: /run/openvswitch + - name: host-run-ovn + hostPath: + path: /run/ovn + - name: cni-conf + hostPath: + path: /etc/cni/net.d + - name: cni-bin + hostPath: + path: /opt/cni/bin + - name: host-ns + hostPath: + path: /var/run/netns + - name: host-dbus + hostPath: + path: /var/run/dbus + - name: host-log-ovs + hostPath: + path: /var/log/openvswitch + - name: kube-ovn-log + hostPath: + path: /var/log/kube-ovn + - name: host-log-ovn + hostPath: + path: /var/log/ovn + - name: localtime + hostPath: + path: /etc/localtime + - name: tmp + hostPath: + path: /tmp + - name: local-bin + hostPath: + path: /usr/local/bin +--- +kind: DaemonSet +apiVersion: apps/v1 +metadata: + name: kube-ovn-pinger + namespace: kube-system + annotations: + kubernetes.io/description: | + This daemon set launches the openvswitch daemon. +spec: + selector: + matchLabels: + app: kube-ovn-pinger + updateStrategy: + type: RollingUpdate + template: + metadata: + labels: + app: kube-ovn-pinger + component: network + type: infra + spec: + priorityClassName: system-node-critical + serviceAccountName: ovn + hostPID: true + containers: + - name: pinger + image: {{ kube_ovn_container_image_repo }}:{{ kube_ovn_container_image_tag }} + command: + - /kube-ovn/kube-ovn-pinger + args: + - --external-address={{ kube_ovn_external_address }}{% if enable_dual_stack_networks %},{{ kube_ovn_external_address_ipv6 }}{% endif %}{{ '' }} + - --external-dns={{ kube_ovn_external_dns }} + - --logtostderr=false + - --alsologtostderr=true + - --log_file=/var/log/kube-ovn/kube-ovn-pinger.log + - --log_file_max_size=0 + imagePullPolicy: {{ k8s_image_pull_policy }} + securityContext: + runAsUser: 0 + privileged: false + env: + - name: ENABLE_SSL + value: "{{ kube_ovn_enable_ssl | lower }}" + - name: POD_IP + valueFrom: + fieldRef: + fieldPath: status.podIP + - name: HOST_IP + valueFrom: + fieldRef: + fieldPath: status.hostIP + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + volumeMounts: + - mountPath: /lib/modules + name: host-modules + readOnly: true + - mountPath: /run/openvswitch + name: host-run-ovs + - mountPath: /var/run/openvswitch + name: host-run-ovs + - mountPath: /var/run/ovn + name: host-run-ovn + - mountPath: /sys + name: host-sys + readOnly: true + - mountPath: /etc/openvswitch + name: host-config-openvswitch + - mountPath: /var/log/openvswitch + name: host-log-ovs + - mountPath: /var/log/ovn + name: host-log-ovn + - mountPath: /var/log/kube-ovn + name: kube-ovn-log + - mountPath: /etc/localtime + name: localtime + - mountPath: /var/run/tls + name: kube-ovn-tls + resources: + requests: + cpu: {{ kube_ovn_pinger_cpu_request }} + memory: {{ kube_ovn_pinger_memory_request }} + limits: + cpu: {{ kube_ovn_pinger_cpu_limit }} + memory: {{ kube_ovn_pinger_memory_limit }} + nodeSelector: + kubernetes.io/os: "linux" + volumes: + - name: host-modules + hostPath: + path: /lib/modules + - name: host-run-ovs + hostPath: + path: /run/openvswitch + - name: host-run-ovn + hostPath: + path: /run/ovn + - name: host-sys + hostPath: + path: /sys + - name: host-config-openvswitch + hostPath: + path: /etc/origin/openvswitch + - name: host-log-ovs + hostPath: + path: /var/log/openvswitch + - name: kube-ovn-log + hostPath: + path: /var/log/kube-ovn + - name: host-log-ovn + hostPath: + path: /var/log/ovn + - name: localtime + hostPath: + path: /etc/localtime + - name: kube-ovn-tls + secret: + optional: true + secretName: kube-ovn-tls +--- +kind: Deployment +apiVersion: apps/v1 +metadata: + name: kube-ovn-monitor + namespace: kube-system + annotations: + kubernetes.io/description: | + Metrics for OVN components: northd, nb and sb. +spec: + replicas: 1 + strategy: + rollingUpdate: + maxSurge: 1 + maxUnavailable: 1 + type: RollingUpdate + selector: + matchLabels: + app: kube-ovn-monitor + template: + metadata: + labels: + app: kube-ovn-monitor + component: network + type: infra + spec: + tolerations: + - effect: NoSchedule + operator: Exists + - key: CriticalAddonsOnly + operator: Exists + affinity: + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchLabels: + app: kube-ovn-monitor + topologyKey: kubernetes.io/hostname + priorityClassName: system-cluster-critical + serviceAccountName: ovn + hostNetwork: true + containers: + - name: kube-ovn-monitor + image: {{ kube_ovn_container_image_repo }}:{{ kube_ovn_container_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + command: ["/kube-ovn/start-ovn-monitor.sh"] + securityContext: + runAsUser: 0 + privileged: false + env: + - name: ENABLE_SSL + value: "{{ kube_ovn_enable_ssl | lower }}" + - name: KUBE_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: POD_IPS + valueFrom: + fieldRef: + fieldPath: status.podIPs + - name: ENABLE_BIND_LOCAL_IP + value: "{{ kube_ovn_bind_local_ip_enabled }}" + resources: + requests: + cpu: {{ kube_ovn_monitor_cpu_request }} + memory: {{ kube_ovn_monitor_memory_request }} + limits: + cpu: {{ kube_ovn_monitor_cpu_limit }} + memory: {{ kube_ovn_monitor_memory_limit }} + volumeMounts: + - mountPath: /var/run/openvswitch + name: host-run-ovs + - mountPath: /var/run/ovn + name: host-run-ovn + - mountPath: /etc/openvswitch + name: host-config-openvswitch + - mountPath: /etc/ovn + name: host-config-ovn + - mountPath: /var/log/openvswitch + name: host-log-ovs + - mountPath: /var/log/ovn + name: host-log-ovn + - mountPath: /etc/localtime + name: localtime + - mountPath: /var/run/tls + name: kube-ovn-tls + readinessProbe: + exec: + command: + - cat + - /var/run/ovn/ovn-controller.pid + periodSeconds: 10 + timeoutSeconds: 45 + livenessProbe: + exec: + command: + - cat + - /var/run/ovn/ovn-controller.pid + initialDelaySeconds: 30 + periodSeconds: 10 + failureThreshold: 5 + timeoutSeconds: 45 + nodeSelector: + kubernetes.io/os: "linux" + kube-ovn/role: "master" + volumes: + - name: host-run-ovs + hostPath: + path: /run/openvswitch + - name: host-run-ovn + hostPath: + path: /run/ovn + - name: host-config-openvswitch + hostPath: + path: /etc/origin/openvswitch + - name: host-config-ovn + hostPath: + path: /etc/origin/ovn + - name: host-log-ovs + hostPath: + path: /var/log/openvswitch + - name: host-log-ovn + hostPath: + path: /var/log/ovn + - name: localtime + hostPath: + path: /etc/localtime + - name: kube-ovn-tls + secret: + optional: true + secretName: kube-ovn-tls +--- +kind: Service +apiVersion: v1 +metadata: + name: kube-ovn-monitor + namespace: kube-system + labels: + app: kube-ovn-monitor +spec: + ports: + - name: metrics + port: 10661 + type: ClusterIP +{% if enable_dual_stack_networks %} + ipFamilyPolicy: PreferDualStack +{% endif %} + selector: + app: kube-ovn-monitor + sessionAffinity: None +--- +kind: Service +apiVersion: v1 +metadata: + name: kube-ovn-pinger + namespace: kube-system + labels: + app: kube-ovn-pinger +spec: +{% if enable_dual_stack_networks %} + ipFamilyPolicy: PreferDualStack +{% endif %} + selector: + app: kube-ovn-pinger + ports: + - port: 8080 + name: metrics +--- +kind: Service +apiVersion: v1 +metadata: + name: kube-ovn-controller + namespace: kube-system + labels: + app: kube-ovn-controller +spec: +{% if enable_dual_stack_networks %} + ipFamilyPolicy: PreferDualStack +{% endif %} + selector: + app: kube-ovn-controller + ports: + - port: 10660 + name: metrics +--- +kind: Service +apiVersion: v1 +metadata: + name: kube-ovn-cni + namespace: kube-system + labels: + app: kube-ovn-cni +spec: +{% if enable_dual_stack_networks %} + ipFamilyPolicy: PreferDualStack +{% endif %} + selector: + app: kube-ovn-cni + ports: + - port: 10665 + name: metrics + {% if kube_ovn_ic_enable %} +--- +kind: ConfigMap +apiVersion: v1 +metadata: + name: ovn-ic-config + namespace: kube-system +data: + enable-ic: "{{ kube_ovn_ic_enable | lower }}" + az-name: "{{ kube_ovn_ic_zone }}" + ic-db-host: "{{ kube_ovn_ic_dbhost }}" + ic-nb-port: "6645" + ic-sb-port: "6646" + gw-nodes: "{{ kube_ovn_central_hosts | join(',') }}" + auto-route: "{{ kube_ovn_ic_autoroute | lower }}" +{% endif %} diff --git a/kubespray/roles/network_plugin/kube-ovn/templates/cni-ovn.yml.j2 b/kubespray/roles/network_plugin/kube-ovn/templates/cni-ovn.yml.j2 new file mode 100644 index 0000000..d632f3b --- /dev/null +++ b/kubespray/roles/network_plugin/kube-ovn/templates/cni-ovn.yml.j2 @@ -0,0 +1,517 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: ovn + namespace: kube-system +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + annotations: + rbac.authorization.k8s.io/system-only: "true" + name: system:ovn +rules: + - apiGroups: + - "kubeovn.io" + resources: + - vpcs + - vpcs/status + - vpc-nat-gateways + - subnets + - subnets/status + - ips + - vips + - vips/status + - vlans + - vlans/status + - provider-networks + - provider-networks/status + - security-groups + - security-groups/status + - iptables-eips + - iptables-fip-rules + - iptables-dnat-rules + - iptables-snat-rules + - iptables-eips/status + - iptables-fip-rules/status + - iptables-dnat-rules/status + - iptables-snat-rules/status + - ovn-eips + - ovn-fips + - ovn-snat-rules + - ovn-eips/status + - ovn-fips/status + - ovn-snat-rules/status + - switch-lb-rules + - switch-lb-rules/status + - vpc-dnses + - vpc-dnses/status + verbs: + - "*" + - apiGroups: + - "" + resources: + - pods + - pods/exec + - namespaces + - nodes + - configmaps + verbs: + - create + - get + - list + - watch + - patch + - update + - apiGroups: + - "k8s.cni.cncf.io" + resources: + - network-attachment-definitions + verbs: + - create + - delete + - get + - list + - update + - apiGroups: + - "" + - networking.k8s.io + - apps + - extensions + resources: + - networkpolicies + - services + - services/status + - endpoints + - statefulsets + - daemonsets + - deployments + - deployments/scale + verbs: + - create + - delete + - update + - patch + - get + - list + - watch + - apiGroups: + - "" + resources: + - events + verbs: + - create + - patch + - update + - apiGroups: + - coordination.k8s.io + resources: + - leases + verbs: + - "*" + - apiGroups: + - "kubevirt.io" + resources: + - virtualmachines + - virtualmachineinstances + verbs: + - get + - list +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: ovn +roleRef: + name: system:ovn + kind: ClusterRole + apiGroup: rbac.authorization.k8s.io +subjects: + - kind: ServiceAccount + name: ovn + namespace: kube-system +--- +kind: Service +apiVersion: v1 +metadata: + name: ovn-nb + namespace: kube-system +spec: + ports: + - name: ovn-nb + protocol: TCP + port: 6641 + targetPort: 6641 + type: ClusterIP +{% if enable_dual_stack_networks %} + ipFamilyPolicy: PreferDualStack +{% endif %} + selector: + app: ovn-central + ovn-nb-leader: "true" + sessionAffinity: None +--- +kind: Service +apiVersion: v1 +metadata: + name: ovn-sb + namespace: kube-system +spec: + ports: + - name: ovn-sb + protocol: TCP + port: 6642 + targetPort: 6642 + type: ClusterIP +{% if enable_dual_stack_networks %} + ipFamilyPolicy: PreferDualStack +{% endif %} + selector: + app: ovn-central + ovn-sb-leader: "true" + sessionAffinity: None +--- +kind: Service +apiVersion: v1 +metadata: + name: ovn-northd + namespace: kube-system +spec: + ports: + - name: ovn-northd + protocol: TCP + port: 6643 + targetPort: 6643 + type: ClusterIP +{% if enable_dual_stack_networks %} + ipFamilyPolicy: PreferDualStack +{% endif %} + selector: + app: ovn-central + ovn-northd-leader: "true" + sessionAffinity: None +--- +kind: Deployment +apiVersion: apps/v1 +metadata: + name: ovn-central + namespace: kube-system + annotations: + kubernetes.io/description: | + OVN components: northd, nb and sb. +spec: + replicas: {{ kube_ovn_central_replics }} + strategy: + rollingUpdate: + maxSurge: 0 + maxUnavailable: 1 + type: RollingUpdate + selector: + matchLabels: + app: ovn-central + template: + metadata: + labels: + app: ovn-central + component: network + type: infra + spec: + tolerations: + - operator: Exists + affinity: + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchLabels: + app: ovn-central + topologyKey: kubernetes.io/hostname + priorityClassName: system-cluster-critical + serviceAccountName: ovn + hostNetwork: true + containers: + - name: ovn-central + image: {{ kube_ovn_container_image_repo }}:{{ kube_ovn_container_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + command: ["/kube-ovn/start-db.sh"] + securityContext: + capabilities: + add: ["SYS_NICE"] + env: + - name: ENABLE_SSL + value: "{{ kube_ovn_enable_ssl | lower }}" + - name: NODE_IPS + value: "{{ kube_ovn_central_ips }}" + - name: POD_IP + valueFrom: + fieldRef: + fieldPath: status.podIP + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + - name: POD_IPS + valueFrom: + fieldRef: + fieldPath: status.podIPs + - name: ENABLE_BIND_LOCAL_IP + value: "{{ kube_ovn_bind_local_ip_enabled }}" + resources: + requests: + cpu: {{ kube_ovn_db_cpu_request }} + memory: {{ kube_ovn_db_memory_request }} + limits: + cpu: {{ kube_ovn_db_cpu_limit }} + memory: {{ kube_ovn_db_memory_limit }} + volumeMounts: + - mountPath: /var/run/openvswitch + name: host-run-ovs + - mountPath: /var/run/ovn + name: host-run-ovn + - mountPath: /sys + name: host-sys + readOnly: true + - mountPath: /etc/openvswitch + name: host-config-openvswitch + - mountPath: /etc/ovn + name: host-config-ovn + - mountPath: /var/log/openvswitch + name: host-log-ovs + - mountPath: /var/log/ovn + name: host-log-ovn + - mountPath: /etc/localtime + name: localtime + - mountPath: /var/run/tls + name: kube-ovn-tls + readinessProbe: + exec: + command: + - bash + - /kube-ovn/ovn-healthcheck.sh + periodSeconds: 15 + timeoutSeconds: 45 + livenessProbe: + exec: + command: + - bash + - /kube-ovn/ovn-healthcheck.sh + initialDelaySeconds: 30 + periodSeconds: 15 + failureThreshold: 5 + timeoutSeconds: 45 + nodeSelector: + kubernetes.io/os: "linux" + kube-ovn/role: "master" + volumes: + - name: host-run-ovs + hostPath: + path: /run/openvswitch + - name: host-run-ovn + hostPath: + path: /run/ovn + - name: host-sys + hostPath: + path: /sys + - name: host-config-openvswitch + hostPath: + path: /etc/origin/openvswitch + - name: host-config-ovn + hostPath: + path: /etc/origin/ovn + - name: host-log-ovs + hostPath: + path: /var/log/openvswitch + - name: host-log-ovn + hostPath: + path: /var/log/ovn + - name: localtime + hostPath: + path: /etc/localtime + - name: kube-ovn-tls + secret: + optional: true + secretName: kube-ovn-tls +--- +kind: DaemonSet +apiVersion: apps/v1 +metadata: + name: ovs-ovn + namespace: kube-system + annotations: + kubernetes.io/description: | + This daemon set launches the openvswitch daemon. +spec: + selector: + matchLabels: + app: ovs + updateStrategy: + type: OnDelete + template: + metadata: + labels: + app: ovs + component: network + type: infra + spec: + tolerations: + - operator: Exists + priorityClassName: system-node-critical + serviceAccountName: ovn + hostNetwork: true + hostPID: true + containers: + - name: openvswitch + image: {% if kube_ovn_dpdk_enabled %}{{ kube_ovn_dpdk_container_image_repo }}:{{ kube_ovn_dpdk_container_image_tag }}{% else %}{{ kube_ovn_container_image_repo }}:{{ kube_ovn_container_image_tag }}{% endif %} + + imagePullPolicy: {{ k8s_image_pull_policy }} + command: [{% if kube_ovn_dpdk_enabled %}"/kube-ovn/start-ovs-dpdk.sh"{% else %}"/kube-ovn/start-ovs.sh"{% endif %}] + securityContext: + runAsUser: 0 + privileged: true + env: + - name: ENABLE_SSL + value: "{{ kube_ovn_enable_ssl | lower }}" + - name: POD_IP + valueFrom: + fieldRef: + fieldPath: status.podIP +{% if not kube_ovn_dpdk_enabled %} + - name: HW_OFFLOAD + value: "{{ kube_ovn_hw_offload | string | lower }}" + - name: TUNNEL_TYPE + value: "{{ kube_ovn_tunnel_type }}" +{% endif %} + - name: KUBE_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: OVN_DB_IPS + value: "{{ kube_ovn_central_ips }}" + volumeMounts: + - mountPath: /var/run/netns + name: host-ns + mountPropagation: HostToContainer + - mountPath: /lib/modules + name: host-modules + readOnly: true + - mountPath: /var/run/openvswitch + name: host-run-ovs + - mountPath: /var/run/ovn + name: host-run-ovn + - mountPath: /sys + name: host-sys + readOnly: true + - mountPath: /etc/cni/net.d + name: cni-conf + - mountPath: /etc/openvswitch + name: host-config-openvswitch + - mountPath: /etc/ovn + name: host-config-ovn + - mountPath: /var/log/openvswitch + name: host-log-ovs + - mountPath: /var/log/ovn + name: host-log-ovn +{% if kube_ovn_dpdk_enabled %} + - mountPath: /opt/ovs-config + name: host-config-ovs + - mountPath: /dev/hugepages + name: hugepage +{% endif %} + - mountPath: /etc/localtime + name: localtime + - mountPath: /var/run/tls + name: kube-ovn-tls + readinessProbe: + exec: + command: + - bash +{% if kube_ovn_dpdk_enabled %} + - /kube-ovn/ovs-dpdk-healthcheck.sh +{% else %} + - /kube-ovn/ovs-healthcheck.sh +{% endif %} + periodSeconds: 5 + timeoutSeconds: 45 + livenessProbe: + exec: + command: + - bash +{% if kube_ovn_dpdk_enabled %} + - /kube-ovn/ovs-dpdk-healthcheck.sh +{% else %} + - /kube-ovn/ovs-healthcheck.sh +{% endif %} + initialDelaySeconds: 60 + periodSeconds: 5 + failureThreshold: 5 + timeoutSeconds: 45 + resources: +{% if kube_ovn_dpdk_enabled %} + requests: + cpu: {{ kube_ovn_dpdk_node_cpu_request }} + memory: {{ kube_ovn_dpdk_node_memory_request }} + limits: + cpu: {{ kube_ovn_dpdk_node_cpu_limit }} + memory: {{ kube_ovn_dpdk_node_memory_limit }} + hugepages-1Gi: 1Gi +{% else %} + requests: + cpu: {{ kube_ovn_node_cpu_request }} + memory: {{ kube_ovn_node_memory_request }} + limits: + cpu: {{ kube_ovn_node_cpu_limit }} + memory: {{ kube_ovn_node_memory_limit }} +{% endif %} + nodeSelector: + kubernetes.io/os: "linux" + volumes: + - name: host-modules + hostPath: + path: /lib/modules + - name: host-run-ovs + hostPath: + path: /run/openvswitch + - name: host-run-ovn + hostPath: + path: /run/ovn + - name: host-sys + hostPath: + path: /sys + - name: host-ns + hostPath: + path: /var/run/netns + - name: cni-conf + hostPath: + path: /etc/cni/net.d + - name: host-config-openvswitch + hostPath: + path: /etc/origin/openvswitch + - name: host-config-ovn + hostPath: + path: /etc/origin/ovn + - name: host-log-ovs + hostPath: + path: /var/log/openvswitch + - name: host-log-ovn + hostPath: + path: /var/log/ovn +{% if kube_ovn_dpdk_enabled %} + - name: host-config-ovs + hostPath: + path: /opt/ovs-config + type: DirectoryOrCreate + - name: hugepage + emptyDir: + medium: HugePages +{% endif %} + - name: localtime + hostPath: + path: /etc/localtime + - name: kube-ovn-tls + secret: + optional: true + secretName: kube-ovn-tls diff --git a/kubespray/roles/network_plugin/kube-router/OWNERS b/kubespray/roles/network_plugin/kube-router/OWNERS new file mode 100644 index 0000000..c40af3c --- /dev/null +++ b/kubespray/roles/network_plugin/kube-router/OWNERS @@ -0,0 +1,6 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +approvers: + - bozzo +reviewers: + - bozzo diff --git a/kubespray/roles/network_plugin/kube-router/defaults/main.yml b/kubespray/roles/network_plugin/kube-router/defaults/main.yml new file mode 100644 index 0000000..5d4dccc --- /dev/null +++ b/kubespray/roles/network_plugin/kube-router/defaults/main.yml @@ -0,0 +1,66 @@ +--- +# Enables Pod Networking -- Advertises and learns the routes to Pods via iBGP +kube_router_run_router: true + +# Enables Network Policy -- sets up iptables to provide ingress firewall for pods +kube_router_run_firewall: true + +# Enables Service Proxy -- sets up IPVS for Kubernetes Services +# see docs/kube-router.md "Caveats" section +kube_router_run_service_proxy: false + +# Add Cluster IP of the service to the RIB so that it gets advertises to the BGP peers. +kube_router_advertise_cluster_ip: false + +# Add External IP of service to the RIB so that it gets advertised to the BGP peers. +kube_router_advertise_external_ip: false + +# Add LoadBalancer IP of service status as set by the LB provider to the RIB so that it gets advertised to the BGP peers. +kube_router_advertise_loadbalancer_ip: false + +# Adjust manifest of kube-router daemonset template with DSR needed changes +kube_router_enable_dsr: false + +# Array of arbitrary extra arguments to kube-router, see +# https://github.com/cloudnativelabs/kube-router/blob/master/docs/user-guide.md +kube_router_extra_args: [] + +# ASN number of the cluster, used when communicating with external BGP routers +kube_router_cluster_asn: ~ + +# ASN numbers of the BGP peer to which cluster nodes will advertise cluster ip and node's pod cidr. +kube_router_peer_router_asns: ~ + +# The ip address of the external router to which all nodes will peer and advertise the cluster ip and pod cidr's. +kube_router_peer_router_ips: ~ + +# The remote port of the external BGP to which all nodes will peer. If not set, default BGP port (179) will be used. +kube_router_peer_router_ports: ~ + +# Setups node CNI to allow hairpin mode, requires node reboots, see +# https://github.com/cloudnativelabs/kube-router/blob/master/docs/user-guide.md#hairpin-mode +kube_router_support_hairpin_mode: false + +# Select DNS Policy ClusterFirstWithHostNet, ClusterFirst, etc. +kube_router_dns_policy: ClusterFirstWithHostNet + +# Adds annotations to kubernetes nodes for advanced configuration of BGP Peers. +# https://github.com/cloudnativelabs/kube-router/blob/master/docs/bgp.md + +# Array of annotations for master +kube_router_annotations_master: [] + +# Array of annotations for every node +kube_router_annotations_node: [] + +# Array of common annotations for every node +kube_router_annotations_all: [] + +# Enables scraping kube-router metrics with Prometheus +kube_router_enable_metrics: false + +# Path to serve Prometheus metrics on +kube_router_metrics_path: /metrics + +# Prometheus metrics port to use +kube_router_metrics_port: 9255 diff --git a/kubespray/roles/network_plugin/kube-router/handlers/main.yml b/kubespray/roles/network_plugin/kube-router/handlers/main.yml new file mode 100644 index 0000000..0723dfd --- /dev/null +++ b/kubespray/roles/network_plugin/kube-router/handlers/main.yml @@ -0,0 +1,24 @@ +--- +- name: Reset_kube_router + command: /bin/true + notify: + - Kube-router | delete kube-router docker containers + - Kube-router | delete kube-router crio/containerd containers + +- name: Kube-router | delete kube-router docker containers + shell: "set -o pipefail && {{ docker_bin_dir }}/docker ps -af name=k8s_POD_kube-router* -q | xargs --no-run-if-empty docker rm -f" + args: + executable: /bin/bash + register: docker_kube_router_remove + until: docker_kube_router_remove is succeeded + retries: 5 + when: container_manager in ["docker"] + +- name: Kube-router | delete kube-router crio/containerd containers + shell: 'set -o pipefail && {{ bin_dir }}/crictl pods --name kube-router* -q | xargs -I% --no-run-if-empty bash -c "{{ bin_dir }}/crictl stopp % && {{ bin_dir }}/crictl rmp %"' + args: + executable: /bin/bash + register: crictl_kube_router_remove + until: crictl_kube_router_remove is succeeded + retries: 5 + when: container_manager in ["crio", "containerd"] diff --git a/kubespray/roles/network_plugin/kube-router/meta/main.yml b/kubespray/roles/network_plugin/kube-router/meta/main.yml new file mode 100644 index 0000000..9b7065f --- /dev/null +++ b/kubespray/roles/network_plugin/kube-router/meta/main.yml @@ -0,0 +1,3 @@ +--- +dependencies: + - role: network_plugin/cni diff --git a/kubespray/roles/network_plugin/kube-router/tasks/annotate.yml b/kubespray/roles/network_plugin/kube-router/tasks/annotate.yml new file mode 100644 index 0000000..67d57a2 --- /dev/null +++ b/kubespray/roles/network_plugin/kube-router/tasks/annotate.yml @@ -0,0 +1,21 @@ +--- +- name: Kube-router | Add annotations on kube_control_plane + command: "{{ kubectl }} annotate --overwrite node {{ ansible_hostname }} {{ item }}" + with_items: + - "{{ kube_router_annotations_master }}" + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: kube_router_annotations_master is defined and inventory_hostname in groups['kube_control_plane'] + +- name: Kube-router | Add annotations on kube_node + command: "{{ kubectl }} annotate --overwrite node {{ ansible_hostname }} {{ item }}" + with_items: + - "{{ kube_router_annotations_node }}" + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: kube_router_annotations_node is defined and inventory_hostname in groups['kube_node'] + +- name: Kube-router | Add common annotations on all servers + command: "{{ kubectl }} annotate --overwrite node {{ ansible_hostname }} {{ item }}" + with_items: + - "{{ kube_router_annotations_all }}" + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: kube_router_annotations_all is defined and inventory_hostname in groups['k8s_cluster'] diff --git a/kubespray/roles/network_plugin/kube-router/tasks/main.yml b/kubespray/roles/network_plugin/kube-router/tasks/main.yml new file mode 100644 index 0000000..b6367f0 --- /dev/null +++ b/kubespray/roles/network_plugin/kube-router/tasks/main.yml @@ -0,0 +1,62 @@ +--- +- name: Kube-router | Create annotations + import_tasks: annotate.yml + tags: annotate + +- name: Kube-router | Create config directory + file: + path: /var/lib/kube-router + state: directory + owner: "{{ kube_owner }}" + recurse: true + mode: 0755 + +- name: Kube-router | Create kubeconfig + template: + src: kubeconfig.yml.j2 + dest: /var/lib/kube-router/kubeconfig + mode: 0644 + owner: "{{ kube_owner }}" + notify: + - Reset_kube_router + +- name: Kube-router | Slurp cni config + slurp: + src: /etc/cni/net.d/10-kuberouter.conflist + register: cni_config_slurp + ignore_errors: true # noqa ignore-errors + +- name: Kube-router | Set cni_config variable + set_fact: + cni_config: "{{ cni_config_slurp.content | b64decode | from_json }}" + when: + - not cni_config_slurp.failed + +- name: Kube-router | Set host_subnet variable + when: + - cni_config is defined + - cni_config | json_query('plugins[?bridge==`kube-bridge`].ipam.subnet') | length > 0 + set_fact: + host_subnet: "{{ cni_config | json_query('plugins[?bridge==`kube-bridge`].ipam.subnet') | first }}" + +- name: Kube-router | Create cni config + template: + src: cni-conf.json.j2 + dest: /etc/cni/net.d/10-kuberouter.conflist + mode: 0644 + owner: "{{ kube_owner }}" + notify: + - Reset_kube_router + +- name: Kube-router | Delete old configuration + file: + path: /etc/cni/net.d/10-kuberouter.conf + state: absent + +- name: Kube-router | Create manifest + template: + src: kube-router.yml.j2 + dest: "{{ kube_config_dir }}/kube-router.yml" + mode: 0644 + delegate_to: "{{ groups['kube_control_plane'] | first }}" + run_once: true diff --git a/kubespray/roles/network_plugin/kube-router/tasks/reset.yml b/kubespray/roles/network_plugin/kube-router/tasks/reset.yml new file mode 100644 index 0000000..ae9ee55 --- /dev/null +++ b/kubespray/roles/network_plugin/kube-router/tasks/reset.yml @@ -0,0 +1,28 @@ +--- +- name: Reset | check kube-dummy-if network device + stat: + path: /sys/class/net/kube-dummy-if + get_attributes: no + get_checksum: no + get_mime: no + register: kube_dummy_if + +- name: Reset | remove the network device created by kube-router + command: ip link del kube-dummy-if + when: kube_dummy_if.stat.exists + +- name: Check kube-bridge exists + stat: + path: /sys/class/net/kube-bridge + get_attributes: no + get_checksum: no + get_mime: no + register: kube_bridge_if + +- name: Reset | donw the network bridge create by kube-router + command: ip link set kube-bridge down + when: kube_bridge_if.stat.exists + +- name: Reset | remove the network bridge create by kube-router + command: ip link del kube-bridge + when: kube_bridge_if.stat.exists diff --git a/kubespray/roles/network_plugin/kube-router/templates/cni-conf.json.j2 b/kubespray/roles/network_plugin/kube-router/templates/cni-conf.json.j2 new file mode 100644 index 0000000..91fafac --- /dev/null +++ b/kubespray/roles/network_plugin/kube-router/templates/cni-conf.json.j2 @@ -0,0 +1,27 @@ +{ + "cniVersion":"0.3.0", + "name":"kubernetes", + "plugins":[ + { + "name":"kubernetes", + "type":"bridge", + "bridge":"kube-bridge", + "isDefaultGateway":true, +{% if kube_router_support_hairpin_mode %} + "hairpinMode":true, +{% endif %} + "ipam":{ +{% if host_subnet is defined %} + "subnet": "{{ host_subnet }}", +{% endif %} + "type":"host-local" + } + }, + { + "type":"portmap", + "capabilities":{ + "portMappings":true + } + } + ] +} diff --git a/kubespray/roles/network_plugin/kube-router/templates/kube-router.yml.j2 b/kubespray/roles/network_plugin/kube-router/templates/kube-router.yml.j2 new file mode 100644 index 0000000..ab677ab --- /dev/null +++ b/kubespray/roles/network_plugin/kube-router/templates/kube-router.yml.j2 @@ -0,0 +1,220 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + labels: + k8s-app: kube-router + tier: node + name: kube-router + namespace: kube-system +spec: + minReadySeconds: 3 + updateStrategy: + rollingUpdate: + maxUnavailable: 1 + type: RollingUpdate + selector: + matchLabels: + k8s-app: kube-router + tier: node + template: + metadata: + labels: + k8s-app: kube-router + tier: node + annotations: +{% if kube_router_enable_metrics %} + prometheus.io/path: {{ kube_router_metrics_path }} + prometheus.io/port: "{{ kube_router_metrics_port }}" + prometheus.io/scrape: "true" +{% endif %} + spec: + priorityClassName: system-node-critical + serviceAccountName: kube-router + containers: + - name: kube-router + image: {{ kube_router_image_repo }}:{{ kube_router_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + args: + - --run-router={{ kube_router_run_router | bool }} + - --run-firewall={{ kube_router_run_firewall | bool }} + - --run-service-proxy={{ kube_router_run_service_proxy | bool }} + - --kubeconfig=/var/lib/kube-router/kubeconfig + - --bgp-graceful-restart=true +{% if kube_router_advertise_cluster_ip %} + - --advertise-cluster-ip +{% endif %} +{% if kube_router_advertise_external_ip %} + - --advertise-external-ip +{% endif %} +{% if kube_router_advertise_loadbalancer_ip %} + - --advertise-loadbalancer-ip +{% endif %} +{% if kube_router_cluster_asn %} + - --cluster-asn={{ kube_router_cluster_asn }} +{% endif %} +{% if kube_router_peer_router_asns %} + - --peer-router-asns={{ kube_router_peer_router_asns }} +{% endif %} +{% if kube_router_peer_router_ips %} + - --peer-router-ips={{ kube_router_peer_router_ips }} +{% endif %} +{% if kube_router_peer_router_ports %} + - --peer-router-ports={{ kube_router_peer_router_ports }} +{% endif %} +{% if kube_router_enable_metrics %} + - --metrics-path={{ kube_router_metrics_path }} + - --metrics-port={{ kube_router_metrics_port }} +{% endif %} +{% if kube_router_enable_dsr %} +{% if container_manager == "docker" %} + - --runtime-endpoint=unix:///var/run/docker.sock +{% endif %} +{% if container_manager == "containerd" %} +{% endif %} + - --runtime-endpoint=unix:///run/containerd/containerd.sock +{% endif %} +{% for arg in kube_router_extra_args %} + - "{{ arg }}" +{% endfor %} + env: + - name: NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: KUBE_ROUTER_CNI_CONF_FILE + value: /etc/cni/net.d/10-kuberouter.conflist + livenessProbe: + httpGet: + path: /healthz + port: 20244 + initialDelaySeconds: 10 + periodSeconds: 3 + resources: + requests: + cpu: 250m + memory: 250Mi + securityContext: + privileged: true + volumeMounts: +{% if kube_router_enable_dsr %} +{% if container_manager == "docker" %} + - name: docker-socket + mountPath: /var/run/docker.sock + readOnly: true +{% endif %} +{% if container_manager == "containerd" %} + - name: containerd-socket + mountPath: /run/containerd/containerd.sock + readOnly: true +{% endif %} +{% endif %} + - name: lib-modules + mountPath: /lib/modules + readOnly: true + - name: cni-conf-dir + mountPath: /etc/cni/net.d + - name: kubeconfig + mountPath: /var/lib/kube-router + readOnly: true + - name: xtables-lock + mountPath: /run/xtables.lock + readOnly: false +{% if kube_router_enable_metrics %} + ports: + - containerPort: {{ kube_router_metrics_port }} + hostPort: {{ kube_router_metrics_port }} + name: metrics + protocol: TCP +{% endif %} + hostNetwork: true + dnsPolicy: {{ kube_router_dns_policy }} +{% if kube_router_enable_dsr %} + hostIPC: true + hostPID: true +{% endif %} + tolerations: + - operator: Exists + volumes: +{% if kube_router_enable_dsr %} +{% if container_manager == "docker" %} + - name: docker-socket + hostPath: + path: /var/run/docker.sock + type: Socket +{% endif %} +{% if container_manager == "containerd" %} + - name: containerd-socket + hostPath: + path: /run/containerd/containerd.sock + type: Socket +{% endif %} +{% endif %} + - name: lib-modules + hostPath: + path: /lib/modules + - name: cni-conf-dir + hostPath: + path: /etc/cni/net.d + - name: kubeconfig + hostPath: + path: /var/lib/kube-router + - name: xtables-lock + hostPath: + path: /run/xtables.lock + type: FileOrCreate + +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: kube-router + namespace: kube-system + +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: kube-router + namespace: kube-system +rules: + - apiGroups: + - "" + resources: + - namespaces + - pods + - services + - nodes + - endpoints + verbs: + - list + - get + - watch + - apiGroups: + - "networking.k8s.io" + resources: + - networkpolicies + verbs: + - list + - get + - watch + - apiGroups: + - extensions + resources: + - networkpolicies + verbs: + - get + - list + - watch +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: kube-router +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: kube-router +subjects: +- kind: ServiceAccount + name: kube-router + namespace: kube-system diff --git a/kubespray/roles/network_plugin/kube-router/templates/kubeconfig.yml.j2 b/kubespray/roles/network_plugin/kube-router/templates/kubeconfig.yml.j2 new file mode 100644 index 0000000..42fd317 --- /dev/null +++ b/kubespray/roles/network_plugin/kube-router/templates/kubeconfig.yml.j2 @@ -0,0 +1,18 @@ +apiVersion: v1 +kind: Config +clusterCIDR: {{ kube_pods_subnet }} +clusters: +- name: cluster + cluster: + certificate-authority: /var/run/secrets/kubernetes.io/serviceaccount/ca.crt + server: {{ kube_apiserver_endpoint }} +users: +- name: kube-router + user: + tokenFile: /var/run/secrets/kubernetes.io/serviceaccount/token +contexts: +- context: + cluster: cluster + user: kube-router + name: kube-router-context +current-context: kube-router-context diff --git a/kubespray/roles/network_plugin/macvlan/OWNERS b/kubespray/roles/network_plugin/macvlan/OWNERS new file mode 100644 index 0000000..c5dfbc7 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/OWNERS @@ -0,0 +1,6 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +approvers: + - simon +reviewers: + - simon diff --git a/kubespray/roles/network_plugin/macvlan/defaults/main.yml b/kubespray/roles/network_plugin/macvlan/defaults/main.yml new file mode 100644 index 0000000..70a8dd0 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/defaults/main.yml @@ -0,0 +1,6 @@ +--- +macvlan_interface: eth0 +enable_nat_default_gateway: true + +# sysctl_file_path to add sysctl conf to +sysctl_file_path: "/etc/sysctl.d/99-sysctl.conf" diff --git a/kubespray/roles/network_plugin/macvlan/files/ifdown-local b/kubespray/roles/network_plugin/macvlan/files/ifdown-local new file mode 100644 index 0000000..003b8a1 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/files/ifdown-local @@ -0,0 +1,6 @@ +#!/bin/bash + +POSTDOWNNAME="/etc/sysconfig/network-scripts/post-down-$1" +if [ -x $POSTDOWNNAME ]; then + exec $POSTDOWNNAME +fi diff --git a/kubespray/roles/network_plugin/macvlan/files/ifdown-macvlan b/kubespray/roles/network_plugin/macvlan/files/ifdown-macvlan new file mode 100755 index 0000000..b79b9c1 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/files/ifdown-macvlan @@ -0,0 +1,40 @@ +#!/bin/bash +# +# initscripts-macvlan +# Copyright (C) 2014 Lars Kellogg-Stedman +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see . + +. /etc/init.d/functions + +cd /etc/sysconfig/network-scripts +. ./network-functions + +[ -f ../network ] && . ../network + +CONFIG=${1} + +need_config ${CONFIG} + +source_config + +OTHERSCRIPT="/etc/sysconfig/network-scripts/ifdown-${REAL_DEVICETYPE}" + +if [ ! -x ${OTHERSCRIPT} ]; then + OTHERSCRIPT="/etc/sysconfig/network-scripts/ifdown-eth" +fi + +${OTHERSCRIPT} ${CONFIG} + +ip link del ${DEVICE} type ${TYPE:-macvlan} diff --git a/kubespray/roles/network_plugin/macvlan/files/ifup-local b/kubespray/roles/network_plugin/macvlan/files/ifup-local new file mode 100755 index 0000000..3b6891e --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/files/ifup-local @@ -0,0 +1,6 @@ +#!/bin/bash + +POSTUPNAME="/etc/sysconfig/network-scripts/post-up-$1" +if [ -x $POSTUPNAME ]; then + exec $POSTUPNAME +fi diff --git a/kubespray/roles/network_plugin/macvlan/files/ifup-macvlan b/kubespray/roles/network_plugin/macvlan/files/ifup-macvlan new file mode 100755 index 0000000..97daec0 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/files/ifup-macvlan @@ -0,0 +1,43 @@ +#!/bin/bash +# +# initscripts-macvlan +# Copyright (C) 2014 Lars Kellogg-Stedman +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see . + +. /etc/init.d/functions + +cd /etc/sysconfig/network-scripts +. ./network-functions + +[ -f ../network ] && . ../network + +CONFIG=${1} + +need_config ${CONFIG} + +source_config + +OTHERSCRIPT="/etc/sysconfig/network-scripts/ifup-${REAL_DEVICETYPE}" + +if [ ! -x ${OTHERSCRIPT} ]; then + OTHERSCRIPT="/etc/sysconfig/network-scripts/ifup-eth" +fi + +ip link add \ + link ${MACVLAN_PARENT} \ + name ${DEVICE} \ + type ${TYPE:-macvlan} mode ${MACVLAN_MODE:-private} + +${OTHERSCRIPT} ${CONFIG} diff --git a/kubespray/roles/network_plugin/macvlan/handlers/main.yml b/kubespray/roles/network_plugin/macvlan/handlers/main.yml new file mode 100644 index 0000000..aba4cbc --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/handlers/main.yml @@ -0,0 +1,20 @@ +--- +- name: Macvlan | restart network + command: /bin/true + notify: + - Macvlan | reload network + when: not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + +- name: Macvlan | reload network + service: + # noqa: jinja[spacing] + name: >- + {% if ansible_os_family == "RedHat" -%} + network + {%- elif ansible_distribution == "Ubuntu" and ansible_distribution_release == "bionic" -%} + systemd-networkd + {%- elif ansible_os_family == "Debian" -%} + networking + {%- endif %} + state: restarted + when: not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] and kube_network_plugin not in ['calico'] diff --git a/kubespray/roles/network_plugin/macvlan/meta/main.yml b/kubespray/roles/network_plugin/macvlan/meta/main.yml new file mode 100644 index 0000000..9b7065f --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/meta/main.yml @@ -0,0 +1,3 @@ +--- +dependencies: + - role: network_plugin/cni diff --git a/kubespray/roles/network_plugin/macvlan/tasks/main.yml b/kubespray/roles/network_plugin/macvlan/tasks/main.yml new file mode 100644 index 0000000..f7c3027 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/tasks/main.yml @@ -0,0 +1,110 @@ +--- +- name: Macvlan | Retrieve Pod Cidr + command: "{{ kubectl }} get nodes {{ kube_override_hostname | default(inventory_hostname) }} -o jsonpath='{.spec.podCIDR}'" + changed_when: false + register: node_pod_cidr_cmd + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: Macvlan | set node_pod_cidr + set_fact: + node_pod_cidr: "{{ node_pod_cidr_cmd.stdout }}" + +- name: Macvlan | Retrieve default gateway network interface + become: false + raw: ip -4 route list 0/0 | sed 's/.*dev \([[:alnum:]]*\).*/\1/' + changed_when: false + register: node_default_gateway_interface_cmd + +- name: Macvlan | set node_default_gateway_interface + set_fact: + node_default_gateway_interface: "{{ node_default_gateway_interface_cmd.stdout | trim }}" + +- name: Macvlan | Install network gateway interface on debian + template: + src: debian-network-macvlan.cfg.j2 + dest: /etc/network/interfaces.d/60-mac0.cfg + mode: 0644 + notify: Macvlan | restart network + when: ansible_os_family in ["Debian"] + +- name: Install macvlan config on RH distros + when: ansible_os_family == "RedHat" + block: + - name: Macvlan | Install macvlan script on centos + copy: + src: "{{ item }}" + dest: /etc/sysconfig/network-scripts/ + owner: root + group: root + mode: "0755" + with_fileglob: + - files/* + + - name: Macvlan | Install post-up script on centos + copy: + src: "files/ifup-local" + dest: /sbin/ + owner: root + group: root + mode: "0755" + when: enable_nat_default_gateway + + - name: Macvlan | Install network gateway interface on centos + template: + src: "{{ item.src }}.j2" + dest: "/etc/sysconfig/network-scripts/{{ item.dst }}" + mode: 0644 + with_items: + - {src: centos-network-macvlan.cfg, dst: ifcfg-mac0 } + - {src: centos-routes-macvlan.cfg, dst: route-mac0 } + - {src: centos-postup-macvlan.cfg, dst: post-up-mac0 } + notify: Macvlan | restart network + +- name: Install macvlan config on Flatcar + when: ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + block: + - name: Macvlan | Install service nat via gateway on Flatcar Container Linux + template: + src: coreos-service-nat_ouside.j2 + dest: /etc/systemd/system/enable_nat_ouside.service + mode: 0644 + when: enable_nat_default_gateway + + - name: Macvlan | Enable service nat via gateway on Flatcar Container Linux + command: "{{ item }}" + with_items: + - systemctl daemon-reload + - systemctl enable enable_nat_ouside.service + when: enable_nat_default_gateway + + - name: Macvlan | Install network gateway interface on Flatcar Container Linux + template: + src: "{{ item.src }}.j2" + dest: "/etc/systemd/network/{{ item.dst }}" + mode: 0644 + with_items: + - {src: coreos-device-macvlan.cfg, dst: macvlan.netdev } + - {src: coreos-interface-macvlan.cfg, dst: output.network } + - {src: coreos-network-macvlan.cfg, dst: macvlan.network } + notify: Macvlan | restart network + +- name: Macvlan | Install cni definition for Macvlan + template: + src: 10-macvlan.conf.j2 + dest: /etc/cni/net.d/10-macvlan.conf + mode: 0644 + +- name: Macvlan | Install loopback definition for Macvlan + template: + src: 99-loopback.conf.j2 + dest: /etc/cni/net.d/99-loopback.conf + mode: 0644 + +- name: Enable net.ipv4.conf.all.arp_notify in sysctl + ansible.posix.sysctl: + name: net.ipv4.conf.all.arp_notify + value: 1 + sysctl_set: yes + sysctl_file: "{{ sysctl_file_path }}" + state: present + reload: yes diff --git a/kubespray/roles/network_plugin/macvlan/templates/10-macvlan.conf.j2 b/kubespray/roles/network_plugin/macvlan/templates/10-macvlan.conf.j2 new file mode 100644 index 0000000..10598a2 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/templates/10-macvlan.conf.j2 @@ -0,0 +1,15 @@ +{ + "cniVersion": "0.4.0", + "name": "mynet", + "type": "macvlan", + "master": "{{ macvlan_interface }}", + "hairpinMode": true, + "ipam": { + "type": "host-local", + "subnet": "{{ node_pod_cidr }}", + "routes": [ + { "dst": "0.0.0.0/0" } + ], + "gateway": "{{ node_pod_cidr|ipaddr('net')|ipaddr(1)|ipaddr('address') }}" + } +} diff --git a/kubespray/roles/network_plugin/macvlan/templates/99-loopback.conf.j2 b/kubespray/roles/network_plugin/macvlan/templates/99-loopback.conf.j2 new file mode 100644 index 0000000..b41ab65 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/templates/99-loopback.conf.j2 @@ -0,0 +1,5 @@ +{ + "cniVersion": "0.2.0", + "name": "lo", + "type": "loopback" +} diff --git a/kubespray/roles/network_plugin/macvlan/templates/centos-network-macvlan.cfg.j2 b/kubespray/roles/network_plugin/macvlan/templates/centos-network-macvlan.cfg.j2 new file mode 100644 index 0000000..a7431c8 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/templates/centos-network-macvlan.cfg.j2 @@ -0,0 +1,13 @@ +DEVICE=mac0 +DEVICETYPE=macvlan +TYPE=macvlan +BOOTPROTO=none +ONBOOT=yes +NM_CONTROLLED=no + +MACVLAN_PARENT={{ macvlan_interface }} +MACVLAN_MODE=bridge + +IPADDR={{ node_pod_cidr|ipaddr('net')|ipaddr(1)|ipaddr('address') }} +NETMASK={{ node_pod_cidr|ipaddr('netmask') }} +NETWORK={{ node_pod_cidr|ipaddr('network') }} diff --git a/kubespray/roles/network_plugin/macvlan/templates/centos-postdown-macvlan.cfg.j2 b/kubespray/roles/network_plugin/macvlan/templates/centos-postdown-macvlan.cfg.j2 new file mode 100644 index 0000000..d62ac2e --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/templates/centos-postdown-macvlan.cfg.j2 @@ -0,0 +1,3 @@ +{% if enable_nat_default_gateway %} +iptables -t nat -D POSTROUTING -s {{ node_pod_cidr|ipaddr('net') }} -o {{ node_default_gateway_interface }} -j MASQUERADE +{% endif %} diff --git a/kubespray/roles/network_plugin/macvlan/templates/centos-postup-macvlan.cfg.j2 b/kubespray/roles/network_plugin/macvlan/templates/centos-postup-macvlan.cfg.j2 new file mode 100644 index 0000000..340bf72 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/templates/centos-postup-macvlan.cfg.j2 @@ -0,0 +1,3 @@ +{% if enable_nat_default_gateway %} +iptables -t nat -I POSTROUTING -s {{ node_pod_cidr|ipaddr('net') }} -o {{ node_default_gateway_interface }} -j MASQUERADE +{% endif %} diff --git a/kubespray/roles/network_plugin/macvlan/templates/centos-routes-macvlan.cfg.j2 b/kubespray/roles/network_plugin/macvlan/templates/centos-routes-macvlan.cfg.j2 new file mode 100644 index 0000000..60400dd --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/templates/centos-routes-macvlan.cfg.j2 @@ -0,0 +1,7 @@ +{% for host in groups['kube_node'] %} +{% if hostvars[host]['access_ip'] is defined %} +{% if hostvars[host]['node_pod_cidr'] != node_pod_cidr %} +{{ hostvars[host]['node_pod_cidr'] }} via {{ hostvars[host]['access_ip'] }} +{% endif %} +{% endif %} +{% endfor %} diff --git a/kubespray/roles/network_plugin/macvlan/templates/coreos-device-macvlan.cfg.j2 b/kubespray/roles/network_plugin/macvlan/templates/coreos-device-macvlan.cfg.j2 new file mode 100644 index 0000000..2418dac --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/templates/coreos-device-macvlan.cfg.j2 @@ -0,0 +1,6 @@ +[NetDev] +Name=mac0 +Kind=macvlan + +[MACVLAN] +Mode=bridge diff --git a/kubespray/roles/network_plugin/macvlan/templates/coreos-interface-macvlan.cfg.j2 b/kubespray/roles/network_plugin/macvlan/templates/coreos-interface-macvlan.cfg.j2 new file mode 100644 index 0000000..342f680 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/templates/coreos-interface-macvlan.cfg.j2 @@ -0,0 +1,6 @@ +[Match] +Name={{ macvlan_interface }} + +[Network] +MACVLAN=mac0 +DHCP=yes diff --git a/kubespray/roles/network_plugin/macvlan/templates/coreos-network-macvlan.cfg.j2 b/kubespray/roles/network_plugin/macvlan/templates/coreos-network-macvlan.cfg.j2 new file mode 100644 index 0000000..ac67389 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/templates/coreos-network-macvlan.cfg.j2 @@ -0,0 +1,17 @@ +[Match] +Name=mac0 + +[Network] +Address={{ node_pod_cidr|ipaddr('net')|ipaddr(1)|ipaddr('address') }}/{{ node_pod_cidr|ipaddr('prefix') }} + +{% for host in groups['kube_node'] %} +{% if hostvars[host]['access_ip'] is defined %} +{% if hostvars[host]['node_pod_cidr'] != node_pod_cidr %} +[Route] +Gateway={{ hostvars[host]['access_ip'] }} +Destination={{ hostvars[host]['node_pod_cidr'] }} +GatewayOnlink=yes + +{% endif %} +{% endif %} +{% endfor %} diff --git a/kubespray/roles/network_plugin/macvlan/templates/coreos-service-nat_ouside.j2 b/kubespray/roles/network_plugin/macvlan/templates/coreos-service-nat_ouside.j2 new file mode 100644 index 0000000..5f00b00 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/templates/coreos-service-nat_ouside.j2 @@ -0,0 +1,6 @@ +[Service] +Type=oneshot +ExecStart=/bin/bash -c "iptables -t nat -I POSTROUTING -s {{ node_pod_cidr|ipaddr('net') }} -o {{ node_default_gateway_interface }} -j MASQUERADE" + +[Install] +WantedBy=sys-subsystem-net-devices-mac0.device diff --git a/kubespray/roles/network_plugin/macvlan/templates/debian-network-macvlan.cfg.j2 b/kubespray/roles/network_plugin/macvlan/templates/debian-network-macvlan.cfg.j2 new file mode 100644 index 0000000..3b3e2e4 --- /dev/null +++ b/kubespray/roles/network_plugin/macvlan/templates/debian-network-macvlan.cfg.j2 @@ -0,0 +1,26 @@ +auto mac0 +iface mac0 inet static + address {{ node_pod_cidr|ipaddr('net')|ipaddr(1)|ipaddr('address') }} + network {{ node_pod_cidr|ipaddr('network') }} + netmask {{ node_pod_cidr|ipaddr('netmask') }} + broadcast {{ node_pod_cidr|ipaddr('broadcast') }} + pre-up ip link add link {{ macvlan_interface }} mac0 type macvlan mode bridge +{% for host in groups['kube_node'] %} +{% if hostvars[host]['access_ip'] is defined %} +{% if hostvars[host]['node_pod_cidr'] != node_pod_cidr %} + post-up ip route add {{ hostvars[host]['node_pod_cidr'] }} via {{ hostvars[host]['access_ip'] }} +{% endif %} +{% endif %} +{% endfor %} +{% if enable_nat_default_gateway %} + post-up iptables -t nat -I POSTROUTING -s {{ node_pod_cidr|ipaddr('net') }} -o {{ node_default_gateway_interface }} -j MASQUERADE +{% endif %} +{% for host in groups['kube_node'] %} +{% if hostvars[host]['access_ip'] is defined %} +{% if hostvars[host]['node_pod_cidr'] != node_pod_cidr %} + post-down ip route del {{ hostvars[host]['node_pod_cidr'] }} via {{ hostvars[host]['access_ip'] }} +{% endif %} +{% endif %} +{% endfor %} + post-down iptables -t nat -D POSTROUTING -s {{ node_pod_cidr|ipaddr('net') }} -o {{ node_default_gateway_interface }} -j MASQUERADE + post-down ip link delete mac0 diff --git a/kubespray/roles/network_plugin/meta/main.yml b/kubespray/roles/network_plugin/meta/main.yml new file mode 100644 index 0000000..dd2c362 --- /dev/null +++ b/kubespray/roles/network_plugin/meta/main.yml @@ -0,0 +1,48 @@ +--- +dependencies: + - role: network_plugin/cni + + - role: network_plugin/cilium + when: kube_network_plugin == 'cilium' or cilium_deploy_additionally | default(false) | bool + tags: + - cilium + + - role: network_plugin/calico + when: kube_network_plugin == 'calico' + tags: + - calico + + - role: network_plugin/flannel + when: kube_network_plugin == 'flannel' + tags: + - flannel + + - role: network_plugin/weave + when: kube_network_plugin == 'weave' + tags: + - weave + + - role: network_plugin/macvlan + when: kube_network_plugin == 'macvlan' + tags: + - macvlan + + - role: network_plugin/kube-ovn + when: kube_network_plugin == 'kube-ovn' + tags: + - kube-ovn + + - role: network_plugin/kube-router + when: kube_network_plugin == 'kube-router' + tags: + - kube-router + + - role: network_plugin/custom_cni + when: kube_network_plugin == 'custom_cni' + tags: + - custom_cni + + - role: network_plugin/multus + when: kube_network_plugin_multus + tags: + - multus diff --git a/kubespray/roles/network_plugin/multus/defaults/main.yml b/kubespray/roles/network_plugin/multus/defaults/main.yml new file mode 100644 index 0000000..c6b7ecd --- /dev/null +++ b/kubespray/roles/network_plugin/multus/defaults/main.yml @@ -0,0 +1,10 @@ +--- +multus_conf_file: "auto" +multus_cni_conf_dir_host: "/etc/cni/net.d" +multus_cni_bin_dir_host: "/opt/cni/bin" +multus_cni_run_dir_host: "/run" +multus_cni_conf_dir: "{{ ('/host', multus_cni_conf_dir_host) | join }}" +multus_cni_bin_dir: "{{ ('/host', multus_cni_bin_dir_host) | join }}" +multus_cni_run_dir: "{{ ('/host', multus_cni_run_dir_host) | join }}" +multus_cni_version: "0.4.0" +multus_kubeconfig_file_host: "{{ (multus_cni_conf_dir_host, '/multus.d/multus.kubeconfig') | join }}" diff --git a/kubespray/roles/network_plugin/multus/files/multus-clusterrole.yml b/kubespray/roles/network_plugin/multus/files/multus-clusterrole.yml new file mode 100644 index 0000000..b574069 --- /dev/null +++ b/kubespray/roles/network_plugin/multus/files/multus-clusterrole.yml @@ -0,0 +1,28 @@ +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: multus +rules: + - apiGroups: ["k8s.cni.cncf.io"] + resources: + - '*' + verbs: + - '*' + - apiGroups: + - "" + resources: + - pods + - pods/status + verbs: + - get + - update + - apiGroups: + - "" + - events.k8s.io + resources: + - events + verbs: + - create + - patch + - update diff --git a/kubespray/roles/network_plugin/multus/files/multus-clusterrolebinding.yml b/kubespray/roles/network_plugin/multus/files/multus-clusterrolebinding.yml new file mode 100644 index 0000000..2d1e1a4 --- /dev/null +++ b/kubespray/roles/network_plugin/multus/files/multus-clusterrolebinding.yml @@ -0,0 +1,13 @@ +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: multus +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: multus +subjects: +- kind: ServiceAccount + name: multus + namespace: kube-system diff --git a/kubespray/roles/network_plugin/multus/files/multus-crd.yml b/kubespray/roles/network_plugin/multus/files/multus-crd.yml new file mode 100644 index 0000000..24b2c58 --- /dev/null +++ b/kubespray/roles/network_plugin/multus/files/multus-crd.yml @@ -0,0 +1,45 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: network-attachment-definitions.k8s.cni.cncf.io +spec: + group: k8s.cni.cncf.io + scope: Namespaced + names: + plural: network-attachment-definitions + singular: network-attachment-definition + kind: NetworkAttachmentDefinition + shortNames: + - net-attach-def + versions: + - name: v1 + served: true + storage: true + schema: + openAPIV3Schema: + description: 'NetworkAttachmentDefinition is a CRD schema specified by the Network Plumbing + Working Group to express the intent for attaching pods to one or more logical or physical + networks. More information available at: https://github.com/k8snetworkplumbingwg/multi-net-spec' + type: object + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this represen + tation of an object. Servers should convert recognized schemas to the + latest internal value, and may reject unrecognized values. More info: + https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: 'NetworkAttachmentDefinition spec defines the desired state of a network attachment' + type: object + properties: + config: + description: 'NetworkAttachmentDefinition config is a JSON-formatted CNI configuration' + type: string diff --git a/kubespray/roles/network_plugin/multus/files/multus-serviceaccount.yml b/kubespray/roles/network_plugin/multus/files/multus-serviceaccount.yml new file mode 100644 index 0000000..6242308 --- /dev/null +++ b/kubespray/roles/network_plugin/multus/files/multus-serviceaccount.yml @@ -0,0 +1,6 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: multus + namespace: kube-system diff --git a/kubespray/roles/network_plugin/multus/meta/main.yml b/kubespray/roles/network_plugin/multus/meta/main.yml new file mode 100644 index 0000000..9b7065f --- /dev/null +++ b/kubespray/roles/network_plugin/multus/meta/main.yml @@ -0,0 +1,3 @@ +--- +dependencies: + - role: network_plugin/cni diff --git a/kubespray/roles/network_plugin/multus/tasks/main.yml b/kubespray/roles/network_plugin/multus/tasks/main.yml new file mode 100644 index 0000000..1428929 --- /dev/null +++ b/kubespray/roles/network_plugin/multus/tasks/main.yml @@ -0,0 +1,36 @@ +--- +- name: Multus | Copy manifest files + copy: + src: "{{ item.file }}" + dest: "{{ kube_config_dir }}" + mode: 0644 + with_items: + - {name: multus-crd, file: multus-crd.yml, type: customresourcedefinition} + - {name: multus-serviceaccount, file: multus-serviceaccount.yml, type: serviceaccount} + - {name: multus-clusterrole, file: multus-clusterrole.yml, type: clusterrole} + - {name: multus-clusterrolebinding, file: multus-clusterrolebinding.yml, type: clusterrolebinding} + register: multus_manifest_1 + when: inventory_hostname == groups['kube_control_plane'][0] + +- name: Multus | Check container engine type + set_fact: + container_manager_types: "{{ ansible_play_hosts_all | map('extract', hostvars, ['container_manager']) | list | unique }}" + +- name: Multus | Copy manifest templates + template: + src: multus-daemonset.yml.j2 + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: multus-daemonset-containerd, file: multus-daemonset-containerd.yml, type: daemonset, engine: containerd } + - {name: multus-daemonset-docker, file: multus-daemonset-docker.yml, type: daemonset, engine: docker } + - {name: multus-daemonset-crio, file: multus-daemonset-crio.yml, type: daemonset, engine: crio } + register: multus_manifest_2 + vars: + query: "*|[?container_manager=='{{ container_manager }}']|[0].inventory_hostname" + vars_from_node: "{{ hostvars | json_query(query) }}" + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: + - item.engine in container_manager_types + - hostvars[inventory_hostname].container_manager == item.engine + - inventory_hostname == vars_from_node diff --git a/kubespray/roles/network_plugin/multus/templates/multus-daemonset.yml.j2 b/kubespray/roles/network_plugin/multus/templates/multus-daemonset.yml.j2 new file mode 100644 index 0000000..10c42c1 --- /dev/null +++ b/kubespray/roles/network_plugin/multus/templates/multus-daemonset.yml.j2 @@ -0,0 +1,79 @@ +--- +kind: DaemonSet +apiVersion: apps/v1 +metadata: +{% if container_manager_types | length >= 2 %} + name: kube-multus-{{ container_manager }}-{{ image_arch }} +{% else %} + name: kube-multus-ds-{{ image_arch }} +{% endif %} + namespace: kube-system + labels: + tier: node + app: multus +spec: + selector: + matchLabels: + tier: node + app: multus + template: + metadata: + labels: + tier: node + app: multus + spec: + hostNetwork: true + dnsPolicy: ClusterFirstWithHostNet + nodeSelector: + kubernetes.io/arch: {{ image_arch }} +{% if container_manager_types | length >= 2 %} + kubespray.io/container_manager: {{ container_manager }} +{% endif %} + tolerations: + - operator: Exists + serviceAccountName: multus + containers: + - name: kube-multus + image: {{ multus_image_repo }}:{{ multus_image_tag }} + command: ["/entrypoint.sh"] + args: + - "--cni-conf-dir={{ multus_cni_conf_dir }}" + - "--cni-bin-dir={{ multus_cni_bin_dir }}" + - "--multus-conf-file={{ multus_conf_file }}" + - "--multus-kubeconfig-file-host={{ multus_kubeconfig_file_host }}" + - "--cni-version={{ multus_cni_version }}" + resources: + requests: + cpu: "100m" + memory: "90Mi" + limits: + cpu: "100m" + memory: "90Mi" + securityContext: + privileged: true +{% if container_manager == 'crio' %} + capabilities: + add: ["SYS_ADMIN"] +{% endif %} + volumeMounts: +{% if container_manager == 'crio' %} + - name: run + mountPath: {{ multus_cni_run_dir }} + mountPropagation: HostToContainer +{% endif %} + - name: cni + mountPath: {{ multus_cni_conf_dir }} + - name: cnibin + mountPath: {{ multus_cni_bin_dir }} + volumes: +{% if container_manager == 'crio' %} + - name: run + hostPath: + path: {{ multus_cni_run_dir_host }} +{% endif %} + - name: cni + hostPath: + path: {{ multus_cni_conf_dir_host }} + - name: cnibin + hostPath: + path: {{ multus_cni_bin_dir_host }} diff --git a/kubespray/roles/network_plugin/ovn4nfv/tasks/main.yml b/kubespray/roles/network_plugin/ovn4nfv/tasks/main.yml new file mode 100644 index 0000000..777fd9a --- /dev/null +++ b/kubespray/roles/network_plugin/ovn4nfv/tasks/main.yml @@ -0,0 +1,16 @@ +--- +- name: Ovn4nfv | Label control-plane node + command: >- + {{ kubectl }} label --overwrite node {{ groups['kube_control_plane'] | first }} ovn4nfv-k8s-plugin=ovn-control-plane + when: + - inventory_hostname == groups['kube_control_plane'][0] + +- name: Ovn4nfv | Create ovn4nfv-k8s manifests + template: + src: "{{ item.file }}.j2" + dest: "{{ kube_config_dir }}/{{ item.file }}" + mode: 0644 + with_items: + - {name: ovn-daemonset, file: ovn-daemonset.yml} + - {name: ovn4nfv-k8s-plugin, file: ovn4nfv-k8s-plugin.yml} + register: ovn4nfv_node_manifests diff --git a/kubespray/roles/network_plugin/weave/defaults/main.yml b/kubespray/roles/network_plugin/weave/defaults/main.yml new file mode 100644 index 0000000..47469ae --- /dev/null +++ b/kubespray/roles/network_plugin/weave/defaults/main.yml @@ -0,0 +1,64 @@ +--- + +# Weave's network password for encryption, if null then no network encryption. +weave_password: ~ + +# If set to 1, disable checking for new Weave Net versions (default is blank, +# i.e. check is enabled) +weave_checkpoint_disable: false + +# Soft limit on the number of connections between peers. Defaults to 100. +weave_conn_limit: 100 + +# Weave Net defaults to enabling hairpin on the bridge side of the veth pair +# for containers attached. If you need to disable hairpin, e.g. your kernel is +# one of those that can panic if hairpin is enabled, then you can disable it by +# setting `HAIRPIN_MODE=false`. +weave_hairpin_mode: true + +# The range of IP addresses used by Weave Net and the subnet they are placed in +# (CIDR format; default 10.32.0.0/12) +weave_ipalloc_range: "{{ kube_pods_subnet }}" + +# Set to 0 to disable Network Policy Controller (default is on) +weave_expect_npc: "{{ enable_network_policy }}" + +# List of addresses of peers in the Kubernetes cluster (default is to fetch the +# list from the api-server) +weave_kube_peers: ~ + +# Set the initialization mode of the IP Address Manager (defaults to consensus +# amongst the KUBE_PEERS) +weave_ipalloc_init: ~ + +# Set the IP address used as a gateway from the Weave network to the host +# network - this is useful if you are configuring the addon as a static pod. +weave_expose_ip: ~ + +# Address and port that the Weave Net daemon will serve Prometheus-style +# metrics on (defaults to 0.0.0.0:6782) +weave_metrics_addr: ~ + +# Address and port that the Weave Net daemon will serve status requests on +# (defaults to disabled) +weave_status_addr: ~ + +# Weave Net defaults to 1376 bytes, but you can set a smaller size if your +# underlying network has a tighter limit, or set a larger size for better +# performance if your network supports jumbo frames (e.g. 8916) +weave_mtu: 1376 + +# Set to 1 to preserve the client source IP address when accessing Service +# annotated with `service.spec.externalTrafficPolicy=Local`. The feature works +# only with Weave IPAM (default). +weave_no_masq_local: true + +# set to nft to use nftables backend for iptables (default is iptables) +weave_iptables_backend: ~ + +# Extra variables that passing to launch.sh, useful for enabling seed mode, see +# https://www.weave.works/docs/net/latest/tasks/ipam/ipam/ +weave_extra_args: ~ + +# Extra variables for weave_npc that passing to launch.sh, useful for change log level, ex --log-level=error +weave_npc_extra_args: ~ diff --git a/kubespray/roles/network_plugin/weave/meta/main.yml b/kubespray/roles/network_plugin/weave/meta/main.yml new file mode 100644 index 0000000..9b7065f --- /dev/null +++ b/kubespray/roles/network_plugin/weave/meta/main.yml @@ -0,0 +1,3 @@ +--- +dependencies: + - role: network_plugin/cni diff --git a/kubespray/roles/network_plugin/weave/tasks/main.yml b/kubespray/roles/network_plugin/weave/tasks/main.yml new file mode 100644 index 0000000..ae4a5a4 --- /dev/null +++ b/kubespray/roles/network_plugin/weave/tasks/main.yml @@ -0,0 +1,12 @@ +--- +- name: Weave | Create manifest + template: + src: weave-net.yml.j2 + dest: "{{ kube_config_dir }}/weave-net.yml" + mode: 0644 + +- name: Weave | Fix nodePort for Weave + template: + src: 10-weave.conflist.j2 + dest: /etc/cni/net.d/10-weave.conflist + mode: 0644 diff --git a/kubespray/roles/network_plugin/weave/templates/10-weave.conflist.j2 b/kubespray/roles/network_plugin/weave/templates/10-weave.conflist.j2 new file mode 100644 index 0000000..9aab7e9 --- /dev/null +++ b/kubespray/roles/network_plugin/weave/templates/10-weave.conflist.j2 @@ -0,0 +1,16 @@ +{ + "cniVersion": "0.3.0", + "name": "weave", + "plugins": [ + { + "name": "weave", + "type": "weave-net", + "hairpinMode": {{ weave_hairpin_mode | bool | lower }} + }, + { + "type": "portmap", + "capabilities": {"portMappings": true}, + "snat": true + } + ] +} diff --git a/kubespray/roles/network_plugin/weave/templates/weave-net.yml.j2 b/kubespray/roles/network_plugin/weave/templates/weave-net.yml.j2 new file mode 100644 index 0000000..3a38865 --- /dev/null +++ b/kubespray/roles/network_plugin/weave/templates/weave-net.yml.j2 @@ -0,0 +1,297 @@ +--- +apiVersion: v1 +kind: List +items: + - apiVersion: v1 + kind: ServiceAccount + metadata: + name: weave-net + labels: + name: weave-net + namespace: kube-system + - apiVersion: rbac.authorization.k8s.io/v1 + kind: ClusterRole + metadata: + name: weave-net + labels: + name: weave-net + rules: + - apiGroups: + - '' + resources: + - pods + - namespaces + - nodes + verbs: + - get + - list + - watch + - apiGroups: + - extensions + resources: + - networkpolicies + verbs: + - get + - list + - watch + - apiGroups: + - 'networking.k8s.io' + resources: + - networkpolicies + verbs: + - get + - list + - watch + - apiGroups: + - '' + resources: + - nodes/status + verbs: + - patch + - update + - apiVersion: rbac.authorization.k8s.io/v1 + kind: ClusterRoleBinding + metadata: + name: weave-net + labels: + name: weave-net + roleRef: + kind: ClusterRole + name: weave-net + apiGroup: rbac.authorization.k8s.io + subjects: + - kind: ServiceAccount + name: weave-net + namespace: kube-system + - apiVersion: rbac.authorization.k8s.io/v1 + kind: Role + metadata: + name: weave-net + namespace: kube-system + labels: + name: weave-net + rules: + - apiGroups: + - '' + resources: + - configmaps + resourceNames: + - weave-net + verbs: + - get + - update + - apiGroups: + - '' + resources: + - configmaps + verbs: + - create + - apiVersion: rbac.authorization.k8s.io/v1 + kind: RoleBinding + metadata: + name: weave-net + namespace: kube-system + labels: + name: weave-net + roleRef: + kind: Role + name: weave-net + apiGroup: rbac.authorization.k8s.io + subjects: + - kind: ServiceAccount + name: weave-net + namespace: kube-system + - apiVersion: apps/v1 + kind: DaemonSet + metadata: + name: weave-net + labels: + name: weave-net + namespace: kube-system + spec: + # Wait 5 seconds to let pod connect before rolling next pod + selector: + matchLabels: + name: weave-net + minReadySeconds: 5 + template: + metadata: + labels: + name: weave-net + spec: + initContainers: + - name: weave-init + image: {{ weave_kube_image_repo }}:{{ weave_kube_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + command: + - /home/weave/init.sh + env: + securityContext: + privileged: true + volumeMounts: + - name: cni-bin + mountPath: /host/opt + - name: cni-bin2 + mountPath: /host/home + - name: cni-conf + mountPath: /host/etc + - name: lib-modules + mountPath: /lib/modules + - name: xtables-lock + mountPath: /run/xtables.lock + readOnly: false + containers: + - name: weave + command: + - /home/weave/launch.sh + env: + - name: INIT_CONTAINER + value: "true" + - name: HOSTNAME + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: spec.nodeName + - name: WEAVE_PASSWORD + valueFrom: + secretKeyRef: + name: weave-net + key: WEAVE_PASSWORD + - name: CHECKPOINT_DISABLE + value: "{{ weave_checkpoint_disable | bool | int }}" + - name: CONN_LIMIT + value: "{{ weave_conn_limit | int }}" + - name: HAIRPIN_MODE + value: "{{ weave_hairpin_mode | bool | lower }}" + - name: IPALLOC_RANGE + value: "{{ weave_ipalloc_range }}" + - name: EXPECT_NPC + value: "{{ weave_expect_npc | bool | int }}" +{% if weave_kube_peers %} + - name: KUBE_PEERS + value: "{{ weave_kube_peers }}" +{% endif %} +{% if weave_ipalloc_init %} + - name: IPALLOC_INIT + value: "{{ weave_ipalloc_init }}" +{% endif %} +{% if weave_expose_ip %} + - name: WEAVE_EXPOSE_IP + value: "{{ weave_expose_ip }}" +{% endif %} +{% if weave_metrics_addr %} + - name: WEAVE_METRICS_ADDR + value: "{{ weave_metrics_addr }}" +{% endif %} +{% if weave_status_addr %} + - name: WEAVE_STATUS_ADDR + value: "{{ weave_status_addr }}" +{% endif %} +{% if weave_iptables_backend %} + - name: IPTABLES_BACKEND + value: "{{ weave_iptables_backend }}" +{% endif %} + - name: WEAVE_MTU + value: "{{ weave_mtu | int }}" + - name: NO_MASQ_LOCAL + value: "{{ weave_no_masq_local | bool | int }}" +{% if weave_extra_args %} + - name: EXTRA_ARGS + value: "{{ weave_extra_args }}" +{% endif %} + image: {{ weave_kube_image_repo }}:{{ weave_kube_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + readinessProbe: + httpGet: + host: 127.0.0.1 + path: /status + port: 6784 + resources: + requests: + cpu: 50m + securityContext: + privileged: true + volumeMounts: + - name: weavedb + mountPath: /weavedb + - name: dbus + mountPath: /host/var/lib/dbus + readOnly: true + - mountPath: /host/etc/machine-id + name: cni-machine-id + readOnly: true + - name: xtables-lock + mountPath: /run/xtables.lock + readOnly: false + - name: weave-npc + env: + - name: HOSTNAME + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: spec.nodeName +{% if weave_npc_extra_args %} + - name: EXTRA_ARGS + value: "{{ weave_npc_extra_args }}" +{% endif %} + image: {{ weave_npc_image_repo }}:{{ weave_npc_image_tag }} + imagePullPolicy: {{ k8s_image_pull_policy }} + resources: + requests: + cpu: 50m + securityContext: + privileged: true + volumeMounts: + - name: xtables-lock + mountPath: /run/xtables.lock + readOnly: false + hostNetwork: true + dnsPolicy: ClusterFirstWithHostNet + hostPID: false + restartPolicy: Always + securityContext: + seLinuxOptions: {} + serviceAccountName: weave-net + tolerations: + - effect: NoSchedule + operator: Exists + - effect: NoExecute + operator: Exists + volumes: + - name: weavedb + hostPath: + path: /var/lib/weave + - name: cni-bin + hostPath: + path: /opt + - name: cni-bin2 + hostPath: + path: /home + - name: cni-conf + hostPath: + path: /etc + - name: cni-machine-id + hostPath: + path: /etc/machine-id + - name: dbus + hostPath: + path: /var/lib/dbus + - name: lib-modules + hostPath: + path: /lib/modules + - name: xtables-lock + hostPath: + path: /run/xtables.lock + type: FileOrCreate + priorityClassName: system-node-critical + updateStrategy: + rollingUpdate: + maxUnavailable: {{ serial | default('20%') }} + type: RollingUpdate + - apiVersion: v1 + kind: Secret + metadata: + name: weave-net + namespace: kube-system + data: + WEAVE_PASSWORD: "{{ weave_password | default("") | b64encode }}" diff --git a/kubespray/roles/recover_control_plane/OWNERS b/kubespray/roles/recover_control_plane/OWNERS new file mode 100644 index 0000000..cb814a1 --- /dev/null +++ b/kubespray/roles/recover_control_plane/OWNERS @@ -0,0 +1,8 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +approvers: + - qvicksilver + - yujunz +reviewers: + - qvicksilver + - yujunz diff --git a/kubespray/roles/recover_control_plane/control-plane/defaults/main.yml b/kubespray/roles/recover_control_plane/control-plane/defaults/main.yml new file mode 100644 index 0000000..229514b --- /dev/null +++ b/kubespray/roles/recover_control_plane/control-plane/defaults/main.yml @@ -0,0 +1,2 @@ +--- +bin_dir: /usr/local/bin diff --git a/kubespray/roles/recover_control_plane/control-plane/tasks/main.yml b/kubespray/roles/recover_control_plane/control-plane/tasks/main.yml new file mode 100644 index 0000000..ec50f3f --- /dev/null +++ b/kubespray/roles/recover_control_plane/control-plane/tasks/main.yml @@ -0,0 +1,29 @@ +--- +- name: Wait for apiserver + command: "{{ kubectl }} get nodes" + environment: + KUBECONFIG: "{{ ansible_env.HOME | default('/root') }}/.kube/config" + register: apiserver_is_ready + until: apiserver_is_ready.rc == 0 + retries: 6 + delay: 10 + changed_when: false + when: groups['broken_kube_control_plane'] + +- name: Delete broken kube_control_plane nodes from cluster + command: "{{ kubectl }} delete node {{ item }}" + environment: + KUBECONFIG: "{{ ansible_env.HOME | default('/root') }}/.kube/config" + with_items: "{{ groups['broken_kube_control_plane'] }}" + register: delete_broken_kube_masters + failed_when: false + when: groups['broken_kube_control_plane'] + +- name: Fail if unable to delete broken kube_control_plane nodes from cluster + fail: + msg: "Unable to delete broken kube_control_plane node: {{ item.item }}" + loop: "{{ delete_broken_kube_masters.results }}" + changed_when: false + when: + - groups['broken_kube_control_plane'] + - "item.rc != 0 and not 'NotFound' in item.stderr" diff --git a/kubespray/roles/recover_control_plane/etcd/tasks/main.yml b/kubespray/roles/recover_control_plane/etcd/tasks/main.yml new file mode 100644 index 0000000..66dbc8b --- /dev/null +++ b/kubespray/roles/recover_control_plane/etcd/tasks/main.yml @@ -0,0 +1,93 @@ +--- +- name: Get etcd endpoint health + command: "{{ bin_dir }}/etcdctl endpoint health" + register: etcd_endpoint_health + ignore_errors: true # noqa ignore-errors + changed_when: false + check_mode: no + environment: + ETCDCTL_API: "3" + ETCDCTL_ENDPOINTS: "{{ etcd_access_addresses }}" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + when: + - groups['broken_etcd'] + +- name: Set healthy fact + set_fact: + healthy: "{{ etcd_endpoint_health.stderr is match('Error: unhealthy cluster') }}" + when: + - groups['broken_etcd'] + +- name: Set has_quorum fact + set_fact: + has_quorum: "{{ etcd_endpoint_health.stdout_lines | select('match', '.*is healthy.*') | list | length >= etcd_endpoint_health.stderr_lines | select('match', '.*is unhealthy.*') | list | length }}" + when: + - groups['broken_etcd'] + +- name: Recover lost etcd quorum + include_tasks: recover_lost_quorum.yml + when: + - groups['broken_etcd'] + - not has_quorum + +- name: Remove etcd data dir + file: + path: "{{ etcd_data_dir }}" + state: absent + delegate_to: "{{ item }}" + with_items: "{{ groups['broken_etcd'] }}" + ignore_errors: true # noqa ignore-errors + when: + - groups['broken_etcd'] + - has_quorum + +- name: Delete old certificates + shell: "rm {{ etcd_cert_dir }}/*{{ item }}*" + with_items: "{{ groups['broken_etcd'] }}" + register: delete_old_cerificates + ignore_errors: true + when: groups['broken_etcd'] + +- name: Fail if unable to delete old certificates + fail: + msg: "Unable to delete old certificates for: {{ item.item }}" + loop: "{{ delete_old_cerificates.results }}" + changed_when: false + when: + - groups['broken_etcd'] + - "item.rc != 0 and not 'No such file or directory' in item.stderr" + +- name: Get etcd cluster members + command: "{{ bin_dir }}/etcdctl member list" + register: member_list + changed_when: false + check_mode: no + environment: + ETCDCTL_API: "3" + ETCDCTL_ENDPOINTS: "{{ etcd_access_addresses }}" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + when: + - groups['broken_etcd'] + - not healthy + - has_quorum + +- name: Remove broken cluster members + command: "{{ bin_dir }}/etcdctl member remove {{ item[1].replace(' ', '').split(',')[0] }}" + environment: + ETCDCTL_API: "3" + ETCDCTL_ENDPOINTS: "{{ etcd_access_addresses }}" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + with_nested: + - "{{ groups['broken_etcd'] }}" + - "{{ member_list.stdout_lines }}" + when: + - groups['broken_etcd'] + - not healthy + - has_quorum + - hostvars[item[0]]['etcd_member_name'] == item[1].replace(' ', '').split(',')[2] diff --git a/kubespray/roles/recover_control_plane/etcd/tasks/recover_lost_quorum.yml b/kubespray/roles/recover_control_plane/etcd/tasks/recover_lost_quorum.yml new file mode 100644 index 0000000..3889628 --- /dev/null +++ b/kubespray/roles/recover_control_plane/etcd/tasks/recover_lost_quorum.yml @@ -0,0 +1,59 @@ +--- +- name: Save etcd snapshot + command: "{{ bin_dir }}/etcdctl snapshot save /tmp/snapshot.db" + environment: + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + ETCDCTL_ENDPOINTS: "{{ etcd_access_addresses.split(',') | first }}" + ETCDCTL_API: "3" + when: etcd_snapshot is not defined + +- name: Transfer etcd snapshot to host + copy: + src: "{{ etcd_snapshot }}" + dest: /tmp/snapshot.db + mode: 0640 + when: etcd_snapshot is defined + +- name: Stop etcd + systemd: + name: etcd + state: stopped + +- name: Remove etcd data-dir + file: + path: "{{ etcd_data_dir }}" + state: absent + +- name: Restore etcd snapshot # noqa command-instead-of-shell + shell: "{{ bin_dir }}/etcdctl snapshot restore /tmp/snapshot.db --name {{ etcd_member_name }} --initial-cluster {{ etcd_member_name }}={{ etcd_peer_url }} --initial-cluster-token k8s_etcd --initial-advertise-peer-urls {{ etcd_peer_url }} --data-dir {{ etcd_data_dir }}" + environment: + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + ETCDCTL_ENDPOINTS: "{{ etcd_access_addresses }}" + ETCDCTL_API: "3" + +- name: Remove etcd snapshot + file: + path: /tmp/snapshot.db + state: absent + +- name: Change etcd data-dir owner + file: + path: "{{ etcd_data_dir }}" + owner: etcd + group: etcd + recurse: true + +- name: Reconfigure etcd + replace: + path: /etc/etcd.env + regexp: "^(ETCD_INITIAL_CLUSTER=).*" + replace: '\1{{ etcd_member_name }}={{ etcd_peer_url }}' + +- name: Start etcd + systemd: + name: etcd + state: started diff --git a/kubespray/roles/recover_control_plane/post-recover/tasks/main.yml b/kubespray/roles/recover_control_plane/post-recover/tasks/main.yml new file mode 100644 index 0000000..a62f912 --- /dev/null +++ b/kubespray/roles/recover_control_plane/post-recover/tasks/main.yml @@ -0,0 +1,20 @@ +--- +# TODO: Figure out why kubeadm does not fix this +- name: Set etcd-servers fact + set_fact: + # noqa: jinja[spacing] + etcd_servers: >- + {% for host in groups['etcd'] -%} + {% if not loop.last -%} + https://{{ hostvars[host].access_ip | default(hostvars[host].ip | default(hostvars[host].ansible_default_ipv4['address'])) }}:2379, + {%- endif -%} + {%- if loop.last -%} + https://{{ hostvars[host].access_ip | default(hostvars[host].ip | default(hostvars[host].ansible_default_ipv4['address'])) }}:2379 + {%- endif -%} + {%- endfor -%} + +- name: Update apiserver etcd-servers list + replace: + path: /etc/kubernetes/manifests/kube-apiserver.yaml + regexp: "(etcd-servers=).*" + replace: "\\1{{ etcd_servers }}" diff --git a/kubespray/roles/remove-node/post-remove/defaults/main.yml b/kubespray/roles/remove-node/post-remove/defaults/main.yml new file mode 100644 index 0000000..11298b9 --- /dev/null +++ b/kubespray/roles/remove-node/post-remove/defaults/main.yml @@ -0,0 +1,3 @@ +--- +delete_node_retries: 10 +delete_node_delay_seconds: 3 diff --git a/kubespray/roles/remove-node/post-remove/tasks/main.yml b/kubespray/roles/remove-node/post-remove/tasks/main.yml new file mode 100644 index 0000000..bc8bfd6 --- /dev/null +++ b/kubespray/roles/remove-node/post-remove/tasks/main.yml @@ -0,0 +1,13 @@ +--- +- name: Remove-node | Delete node + command: "{{ kubectl }} delete node {{ kube_override_hostname | default(inventory_hostname) }}" + delegate_to: "{{ groups['kube_control_plane'] | first }}" + when: + - groups['kube_control_plane'] | length > 0 + # ignore servers that are not nodes + - inventory_hostname in groups['k8s_cluster'] and kube_override_hostname | default(inventory_hostname) in nodes.stdout_lines + retries: "{{ delete_node_retries }}" + # Sometimes the api-server can have a short window of indisponibility when we delete a master node + delay: "{{ delete_node_delay_seconds }}" + register: result + until: result is not failed diff --git a/kubespray/roles/remove-node/pre-remove/defaults/main.yml b/kubespray/roles/remove-node/pre-remove/defaults/main.yml new file mode 100644 index 0000000..deaa8af --- /dev/null +++ b/kubespray/roles/remove-node/pre-remove/defaults/main.yml @@ -0,0 +1,6 @@ +--- +allow_ungraceful_removal: false +drain_grace_period: 300 +drain_timeout: 360s +drain_retries: 3 +drain_retry_delay_seconds: 10 diff --git a/kubespray/roles/remove-node/pre-remove/tasks/main.yml b/kubespray/roles/remove-node/pre-remove/tasks/main.yml new file mode 100644 index 0000000..6f6c314 --- /dev/null +++ b/kubespray/roles/remove-node/pre-remove/tasks/main.yml @@ -0,0 +1,43 @@ +--- +- name: Remove-node | List nodes + command: >- + {{ kubectl }} get nodes -o go-template={% raw %}'{{ range .items }}{{ .metadata.name }}{{ "\n" }}{{ end }}'{% endraw %} + register: nodes + when: + - groups['kube_control_plane'] | length > 0 + delegate_to: "{{ groups['kube_control_plane'] | first }}" + changed_when: false + run_once: true + +- name: Remove-node | Drain node except daemonsets resource + command: >- + {{ kubectl }} drain + --force + --ignore-daemonsets + --grace-period {{ drain_grace_period }} + --timeout {{ drain_timeout }} + --delete-emptydir-data {{ kube_override_hostname | default(inventory_hostname) }} + when: + - groups['kube_control_plane'] | length > 0 + # ignore servers that are not nodes + - kube_override_hostname | default(inventory_hostname) in nodes.stdout_lines + register: result + failed_when: result.rc != 0 and not allow_ungraceful_removal + delegate_to: "{{ groups['kube_control_plane'] | first }}" + until: result.rc == 0 or allow_ungraceful_removal + retries: "{{ drain_retries }}" + delay: "{{ drain_retry_delay_seconds }}" + +- name: Remove-node | Wait until Volumes will be detached from the node + command: >- + {{ kubectl }} get volumeattachments -o go-template={% raw %}'{{ range .items }}{{ .spec.nodeName }}{{ "\n" }}{{ end }}'{% endraw %} + register: nodes_with_volumes + delegate_to: "{{ groups['kube_control_plane'] | first }}" + changed_when: false + until: not (kube_override_hostname | default(inventory_hostname) in nodes_with_volumes.stdout_lines) + retries: 3 + delay: "{{ drain_grace_period }}" + when: + - groups['kube_control_plane'] | length > 0 + - not allow_ungraceful_removal + - kube_override_hostname | default(inventory_hostname) in nodes.stdout_lines diff --git a/kubespray/roles/remove-node/remove-etcd-node/tasks/main.yml b/kubespray/roles/remove-node/remove-etcd-node/tasks/main.yml new file mode 100644 index 0000000..0279018 --- /dev/null +++ b/kubespray/roles/remove-node/remove-etcd-node/tasks/main.yml @@ -0,0 +1,58 @@ +--- +- name: Lookup node IP in kubernetes + command: > + {{ kubectl }} get nodes {{ node }} + -o jsonpath='{range .status.addresses[?(@.type=="InternalIP")]}{@.address}{"\n"}{end}' + register: remove_node_ip + when: + - groups['kube_control_plane'] | length > 0 + - inventory_hostname in groups['etcd'] + - ip is not defined + - access_ip is not defined + delegate_to: "{{ groups['etcd'] | first }}" + failed_when: false + +- name: Set node IP + set_fact: + node_ip: "{{ ip | default(access_ip | default(remove_node_ip.stdout)) | trim }}" + when: + - inventory_hostname in groups['etcd'] + +- name: Make sure node_ip is set + assert: + that: node_ip is defined and node_ip | length > 0 + msg: "Etcd node ip is not set !" + when: + - inventory_hostname in groups['etcd'] + +- name: Lookup etcd member id + shell: "set -o pipefail && {{ bin_dir }}/etcdctl member list | grep {{ node_ip }} | cut -d, -f1" + args: + executable: /bin/bash + register: etcd_member_id + ignore_errors: true # noqa ignore-errors + changed_when: false + check_mode: no + tags: + - facts + environment: + ETCDCTL_API: "3" + ETCDCTL_CERT: "{{ kube_cert_dir + '/etcd/server.crt' if etcd_deployment_type == 'kubeadm' else etcd_cert_dir + '/admin-' + groups['etcd'] | first + '.pem' }}" + ETCDCTL_KEY: "{{ kube_cert_dir + '/etcd/server.key' if etcd_deployment_type == 'kubeadm' else etcd_cert_dir + '/admin-' + groups['etcd'] | first + '-key.pem' }}" + ETCDCTL_CACERT: "{{ kube_cert_dir + '/etcd/ca.crt' if etcd_deployment_type == 'kubeadm' else etcd_cert_dir + '/ca.pem' }}" + ETCDCTL_ENDPOINTS: "https://127.0.0.1:2379" + delegate_to: "{{ groups['etcd'] | first }}" + when: inventory_hostname in groups['etcd'] + +- name: Remove etcd member from cluster + command: "{{ bin_dir }}/etcdctl member remove {{ etcd_member_id.stdout }}" + environment: + ETCDCTL_API: "3" + ETCDCTL_CERT: "{{ kube_cert_dir + '/etcd/server.crt' if etcd_deployment_type == 'kubeadm' else etcd_cert_dir + '/admin-' + groups['etcd'] | first + '.pem' }}" + ETCDCTL_KEY: "{{ kube_cert_dir + '/etcd/server.key' if etcd_deployment_type == 'kubeadm' else etcd_cert_dir + '/admin-' + groups['etcd'] | first + '-key.pem' }}" + ETCDCTL_CACERT: "{{ kube_cert_dir + '/etcd/ca.crt' if etcd_deployment_type == 'kubeadm' else etcd_cert_dir + '/ca.pem' }}" + ETCDCTL_ENDPOINTS: "https://127.0.0.1:2379" + delegate_to: "{{ groups['etcd'] | first }}" + when: + - inventory_hostname in groups['etcd'] + - etcd_member_id.stdout | length > 0 diff --git a/kubespray/roles/reset/defaults/main.yml b/kubespray/roles/reset/defaults/main.yml new file mode 100644 index 0000000..3d58dd9 --- /dev/null +++ b/kubespray/roles/reset/defaults/main.yml @@ -0,0 +1,18 @@ +--- +flush_iptables: true +reset_restart_network: true + +reset_restart_network_service_name: >- + {% if ansible_os_family == "RedHat" -%} + {%- + if ansible_distribution_major_version | int >= 8 + or is_fedora_coreos or ansible_distribution == "Fedora" -%} + NetworkManager + {%- else -%} + network + {%- endif -%} + {%- elif ansible_distribution == "Ubuntu" -%} + systemd-networkd + {%- elif ansible_os_family == "Debian" -%} + networking + {%- endif %} diff --git a/kubespray/roles/reset/tasks/main.yml b/kubespray/roles/reset/tasks/main.yml new file mode 100644 index 0000000..198b2c4 --- /dev/null +++ b/kubespray/roles/reset/tasks/main.yml @@ -0,0 +1,439 @@ +--- +- name: Reset | stop services + service: + name: "{{ item }}" + state: stopped + with_items: + - kubelet.service + - cri-dockerd.service + - cri-dockerd.socket + failed_when: false + tags: + - services + +- name: Reset | remove services + file: + path: "/etc/systemd/system/{{ item }}" + state: absent + with_items: + - kubelet.service + - cri-dockerd.service + - cri-dockerd.socket + - calico-node.service + - containerd.service.d/http-proxy.conf + - crio.service.d/http-proxy.conf + - k8s-certs-renew.service + - k8s-certs-renew.timer + register: services_removed + tags: + - services + - containerd + - crio + +- name: Reset | Remove Docker + include_role: + name: container-engine/docker + tasks_from: reset + when: container_manager == 'docker' + tags: + - docker + +- name: Reset | systemctl daemon-reload # noqa no-handler + systemd: + daemon_reload: true + when: services_removed.changed + +- name: Reset | check if crictl is present + stat: + path: "{{ bin_dir }}/crictl" + get_attributes: no + get_checksum: no + get_mime: no + register: crictl + +- name: Reset | stop all cri containers + shell: "set -o pipefail && {{ bin_dir }}/crictl ps -q | xargs -r {{ bin_dir }}/crictl -t 60s stop" + args: + executable: /bin/bash + register: remove_all_cri_containers + retries: 5 + until: remove_all_cri_containers.rc == 0 + delay: 5 + tags: + - crio + - containerd + when: + - crictl.stat.exists + - container_manager in ["crio", "containerd"] + - ansible_facts.services['containerd.service'] is defined or ansible_facts.services['cri-o.service'] is defined + ignore_errors: true # noqa ignore-errors + +- name: Reset | force remove all cri containers + command: "{{ bin_dir }}/crictl rm -a -f" + register: remove_all_cri_containers + retries: 5 + until: remove_all_cri_containers.rc == 0 + delay: 5 + tags: + - crio + - containerd + when: + - crictl.stat.exists + - container_manager in ["crio", "containerd"] + - deploy_container_engine + - ansible_facts.services['containerd.service'] is defined or ansible_facts.services['cri-o.service'] is defined + ignore_errors: true # noqa ignore-errors + +- name: Reset | stop and disable crio service + service: + name: crio + state: stopped + enabled: false + failed_when: false + tags: [ crio ] + when: container_manager == "crio" + +- name: Reset | forcefully wipe CRI-O's container and image storage + command: "crio wipe -f" + failed_when: false + tags: [ crio ] + when: container_manager == "crio" + +- name: Reset | stop all cri pods + shell: "set -o pipefail && {{ bin_dir }}/crictl pods -q | xargs -r {{ bin_dir }}/crictl -t 60s stopp" + args: + executable: /bin/bash + register: remove_all_cri_containers + retries: 5 + until: remove_all_cri_containers.rc == 0 + delay: 5 + tags: [ containerd ] + when: + - crictl.stat.exists + - container_manager == "containerd" + - ansible_facts.services['containerd.service'] is defined or ansible_facts.services['cri-o.service'] is defined + ignore_errors: true # noqa ignore-errors + +- name: Reset | force remove all cri pods + block: + - name: Reset | force remove all cri pods + command: "{{ bin_dir }}/crictl rmp -a -f" + register: remove_all_cri_containers + retries: 5 + until: remove_all_cri_containers.rc == 0 + delay: 5 + tags: [ containerd ] + when: + - crictl.stat.exists + - container_manager == "containerd" + - ansible_facts.services['containerd.service'] is defined or ansible_facts.services['cri-o.service'] is defined + + rescue: + - name: Reset | force remove all cri pods (rescue) + shell: "ip netns list | cut -d' ' -f 1 | xargs -n1 ip netns delete && {{ bin_dir }}/crictl rmp -a -f" + ignore_errors: true # noqa ignore-errors + changed_when: true + +- name: Reset | stop etcd services + service: + name: "{{ item }}" + state: stopped + with_items: + - etcd + - etcd-events + failed_when: false + tags: + - services + +- name: Reset | remove etcd services + file: + path: "/etc/systemd/system/{{ item }}.service" + state: absent + with_items: + - etcd + - etcd-events + register: services_removed + tags: + - services + +- name: Reset | remove containerd + when: container_manager == 'containerd' + block: + - name: Reset | stop containerd service + service: + name: containerd + state: stopped + failed_when: false + tags: + - services + + - name: Reset | remove containerd service + file: + path: /etc/systemd/system/containerd.service + state: absent + register: services_removed + tags: + - services + +- name: Reset | gather mounted kubelet dirs + shell: set -o pipefail && mount | grep /var/lib/kubelet/ | awk '{print $3}' | tac + args: + executable: /bin/bash + check_mode: no + register: mounted_dirs + failed_when: false + changed_when: false + tags: + - mounts + +- name: Reset | unmount kubelet dirs + command: umount -f {{ item }} + with_items: "{{ mounted_dirs.stdout_lines }}" + register: umount_dir + when: mounted_dirs + retries: 4 + until: umount_dir.rc == 0 + delay: 5 + tags: + - mounts + +- name: Flush iptables + iptables: + table: "{{ item }}" + flush: yes + with_items: + - filter + - nat + - mangle + - raw + when: flush_iptables | bool + tags: + - iptables + +- name: Flush ip6tables + iptables: + table: "{{ item }}" + flush: yes + ip_version: ipv6 + with_items: + - filter + - nat + - mangle + - raw + when: flush_iptables | bool and enable_dual_stack_networks + tags: + - ip6tables + +- name: Clear IPVS virtual server table + command: "ipvsadm -C" + ignore_errors: true # noqa ignore-errors + when: + - kube_proxy_mode == 'ipvs' and inventory_hostname in groups['k8s_cluster'] + +- name: Reset | check kube-ipvs0 network device + stat: + path: /sys/class/net/kube-ipvs0 + get_attributes: no + get_checksum: no + get_mime: no + register: kube_ipvs0 + +- name: Reset | Remove kube-ipvs0 + command: "ip link del kube-ipvs0" + when: + - kube_proxy_mode == 'ipvs' + - kube_ipvs0.stat.exists + +- name: Reset | check nodelocaldns network device + stat: + path: /sys/class/net/nodelocaldns + get_attributes: no + get_checksum: no + get_mime: no + register: nodelocaldns_device + +- name: Reset | Remove nodelocaldns + command: "ip link del nodelocaldns" + when: + - enable_nodelocaldns | default(false) | bool + - nodelocaldns_device.stat.exists + +- name: Reset | Check whether /var/lib/kubelet directory exists + stat: + path: /var/lib/kubelet + get_attributes: no + get_checksum: no + get_mime: no + register: var_lib_kubelet_directory + +- name: Reset | Find files/dirs with immutable flag in /var/lib/kubelet + command: lsattr -laR /var/lib/kubelet + become: true + register: var_lib_kubelet_files_dirs_w_attrs + changed_when: false + no_log: true + when: var_lib_kubelet_directory.stat.exists + +- name: Reset | Remove immutable flag from files/dirs in /var/lib/kubelet + file: + path: "{{ filedir_path }}" + state: touch + attributes: "-i" + mode: 0644 + loop: "{{ var_lib_kubelet_files_dirs_w_attrs.stdout_lines | select('search', 'Immutable') | list }}" + loop_control: + loop_var: file_dir_line + label: "{{ filedir_path }}" + vars: + filedir_path: "{{ file_dir_line.split(' ')[0] }}" + when: var_lib_kubelet_directory.stat.exists + +- name: Reset | delete some files and directories + file: + path: "{{ item }}" + state: absent + with_items: + - "{{ kube_config_dir }}" + - /var/lib/kubelet + - "{{ containerd_storage_dir }}" + - "{{ ansible_env.HOME | default('/root') }}/.kube" + - "{{ ansible_env.HOME | default('/root') }}/.helm" + - "{{ ansible_env.HOME | default('/root') }}/.config/helm" + - "{{ ansible_env.HOME | default('/root') }}/.cache/helm" + - "{{ ansible_env.HOME | default('/root') }}/.local/share/helm" + - "{{ etcd_data_dir }}" + - "{{ etcd_events_data_dir }}" + - "{{ etcd_config_dir }}" + - /var/log/calico + - /var/log/openvswitch + - /var/log/ovn + - /var/log/kube-ovn + - /etc/cni + - /etc/nerdctl + - "{{ nginx_config_dir }}" + - /etc/dnsmasq.d + - /etc/dnsmasq.conf + - /etc/dnsmasq.d-available + - /etc/etcd.env + - /etc/calico + - /etc/NetworkManager/conf.d/calico.conf + - /etc/NetworkManager/conf.d/k8s.conf + - /etc/weave.env + - /opt/cni + - /etc/dhcp/dhclient.d/zdnsupdate.sh + - /etc/dhcp/dhclient-exit-hooks.d/zdnsupdate + - /run/flannel + - /etc/flannel + - /run/kubernetes + - /usr/local/share/ca-certificates/etcd-ca.crt + - /usr/local/share/ca-certificates/kube-ca.crt + - /etc/ssl/certs/etcd-ca.pem + - /etc/ssl/certs/kube-ca.pem + - /etc/pki/ca-trust/source/anchors/etcd-ca.crt + - /etc/pki/ca-trust/source/anchors/kube-ca.crt + - /var/log/pods/ + - "{{ bin_dir }}/kubelet" + - "{{ bin_dir }}/cri-dockerd" + - "{{ bin_dir }}/etcd-scripts" + - "{{ bin_dir }}/etcd" + - "{{ bin_dir }}/etcd-events" + - "{{ bin_dir }}/etcdctl" + - "{{ bin_dir }}/etcdctl.sh" + - "{{ bin_dir }}/kubernetes-scripts" + - "{{ bin_dir }}/kubectl" + - "{{ bin_dir }}/kubeadm" + - "{{ bin_dir }}/helm" + - "{{ bin_dir }}/calicoctl" + - "{{ bin_dir }}/calicoctl.sh" + - "{{ bin_dir }}/calico-upgrade" + - "{{ bin_dir }}/weave" + - "{{ bin_dir }}/crictl" + - "{{ bin_dir }}/nerdctl" + - "{{ bin_dir }}/netctl" + - "{{ bin_dir }}/k8s-certs-renew.sh" + - /var/lib/cni + - /etc/openvswitch + - /run/openvswitch + - /var/lib/kube-router + - /var/lib/calico + - /etc/cilium + - /run/calico + - /etc/bash_completion.d/kubectl.sh + - /etc/bash_completion.d/crictl + - /etc/bash_completion.d/nerdctl + - /etc/bash_completion.d/krew + - /etc/bash_completion.d/krew.sh + - "{{ krew_root_dir }}" + - /etc/modules-load.d/kube_proxy-ipvs.conf + - /etc/modules-load.d/kubespray-br_netfilter.conf + - /etc/modules-load.d/kubespray-kata-containers.conf + - /usr/libexec/kubernetes + - /etc/origin/openvswitch + - /etc/origin/ovn + - "{{ sysctl_file_path }}" + - /etc/crictl.yaml + ignore_errors: true # noqa ignore-errors + tags: + - files + +- name: Reset | remove containerd binary files + file: + path: "{{ containerd_bin_dir }}/{{ item }}" + state: absent + with_items: + - containerd + - containerd-shim + - containerd-shim-runc-v1 + - containerd-shim-runc-v2 + - containerd-stress + - crictl + - critest + - ctd-decoder + - ctr + - runc + ignore_errors: true # noqa ignore-errors + when: container_manager == 'containerd' + tags: + - files + +- name: Reset | remove dns settings from dhclient.conf + blockinfile: + path: "{{ item }}" + state: absent + marker: "# Ansible entries {mark}" + failed_when: false + with_items: + - /etc/dhclient.conf + - /etc/dhcp/dhclient.conf + tags: + - files + - dns + +- name: Reset | remove host entries from /etc/hosts + blockinfile: + path: "/etc/hosts" + state: absent + marker: "# Ansible inventory hosts {mark}" + tags: + - files + - dns + +- name: Reset | include file with reset tasks specific to the network_plugin if exists + include_role: + name: "network_plugin/{{ kube_network_plugin }}" + tasks_from: reset + when: + - kube_network_plugin in ['flannel', 'cilium', 'kube-router', 'calico'] + tags: + - network + +- name: Reset | Restart network + service: + name: "{{ reset_restart_network_service_name }}" + state: restarted + when: + - ansible_os_family not in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + - reset_restart_network + tags: + - services + - network diff --git a/kubespray/roles/upgrade/post-upgrade/defaults/main.yml b/kubespray/roles/upgrade/post-upgrade/defaults/main.yml new file mode 100644 index 0000000..aa72843 --- /dev/null +++ b/kubespray/roles/upgrade/post-upgrade/defaults/main.yml @@ -0,0 +1,5 @@ +--- +# how long to wait for cilium after upgrade before uncordoning +upgrade_post_cilium_wait_timeout: 120s +upgrade_node_post_upgrade_confirm: false +upgrade_node_post_upgrade_pause_seconds: 0 diff --git a/kubespray/roles/upgrade/post-upgrade/tasks/main.yml b/kubespray/roles/upgrade/post-upgrade/tasks/main.yml new file mode 100644 index 0000000..434ef1e --- /dev/null +++ b/kubespray/roles/upgrade/post-upgrade/tasks/main.yml @@ -0,0 +1,32 @@ +--- +- name: Wait for cilium + when: + - needs_cordoning | default(false) + - kube_network_plugin == 'cilium' + command: > + {{ kubectl }} + wait pod -n kube-system -l k8s-app=cilium + --field-selector 'spec.nodeName=={{ kube_override_hostname | default(inventory_hostname) }}' + --for=condition=Ready + --timeout={{ upgrade_post_cilium_wait_timeout }} + delegate_to: "{{ groups['kube_control_plane'][0] }}" + +- name: Confirm node uncordon + pause: + echo: yes + prompt: "Ready to uncordon node?" + when: + - upgrade_node_post_upgrade_confirm + +- name: Wait before uncordoning node + pause: + seconds: "{{ upgrade_node_post_upgrade_pause_seconds }}" + when: + - not upgrade_node_post_upgrade_confirm + - upgrade_node_post_upgrade_pause_seconds != 0 + +- name: Uncordon node + command: "{{ kubectl }} uncordon {{ kube_override_hostname | default(inventory_hostname) }}" + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: + - needs_cordoning | default(false) diff --git a/kubespray/roles/upgrade/pre-upgrade/defaults/main.yml b/kubespray/roles/upgrade/pre-upgrade/defaults/main.yml new file mode 100644 index 0000000..900b834 --- /dev/null +++ b/kubespray/roles/upgrade/pre-upgrade/defaults/main.yml @@ -0,0 +1,20 @@ +--- +drain_grace_period: 300 +drain_timeout: 360s +drain_pod_selector: "" +drain_nodes: true +drain_retries: 3 +drain_retry_delay_seconds: 10 + +drain_fallback_enabled: false +drain_fallback_grace_period: 300 +drain_fallback_timeout: 360s +drain_fallback_retries: 0 +drain_fallback_retry_delay_seconds: 10 + +upgrade_node_always_cordon: false +upgrade_node_uncordon_after_drain_failure: true +upgrade_node_fail_if_drain_fails: true + +upgrade_node_confirm: false +upgrade_node_pause_seconds: 0 diff --git a/kubespray/roles/upgrade/pre-upgrade/tasks/main.yml b/kubespray/roles/upgrade/pre-upgrade/tasks/main.yml new file mode 100644 index 0000000..58dfee0 --- /dev/null +++ b/kubespray/roles/upgrade/pre-upgrade/tasks/main.yml @@ -0,0 +1,131 @@ +--- +# Wait for upgrade +- name: Confirm node upgrade + pause: + echo: yes + prompt: "Ready to upgrade node? (Press Enter to continue or Ctrl+C for other options)" + when: + - upgrade_node_confirm + +- name: Wait before upgrade node + pause: + seconds: "{{ upgrade_node_pause_seconds }}" + when: + - not upgrade_node_confirm + - upgrade_node_pause_seconds != 0 + +# Node Ready: type = ready, status = True +# Node NotReady: type = ready, status = Unknown +- name: See if node is in ready state + command: > + {{ kubectl }} get node {{ kube_override_hostname | default(inventory_hostname) }} + -o jsonpath='{ range .status.conditions[?(@.type == "Ready")].status }{ @ }{ end }' + register: kubectl_node_ready + delegate_to: "{{ groups['kube_control_plane'][0] }}" + failed_when: false + changed_when: false + +# SchedulingDisabled: unschedulable = true +# else unschedulable key doesn't exist +- name: See if node is schedulable + command: > + {{ kubectl }} get node {{ kube_override_hostname | default(inventory_hostname) }} + -o jsonpath='{ .spec.unschedulable }' + register: kubectl_node_schedulable + delegate_to: "{{ groups['kube_control_plane'][0] }}" + failed_when: false + changed_when: false + +- name: Set if node needs cordoning + set_fact: + # noqa: jinja[spacing] + needs_cordoning: >- + {% if (kubectl_node_ready.stdout == "True" and not kubectl_node_schedulable.stdout) or upgrade_node_always_cordon -%} + true + {%- else -%} + false + {%- endif %} + +- name: Node draining + delegate_to: "{{ groups['kube_control_plane'][0] }}" + when: + - needs_cordoning + block: + - name: Cordon node + command: "{{ kubectl }} cordon {{ kube_override_hostname | default(inventory_hostname) }}" + delegate_to: "{{ groups['kube_control_plane'][0] }}" + changed_when: true + + - name: Check kubectl version + command: "{{ kubectl }} version --client --short" + register: kubectl_version + delegate_to: "{{ groups['kube_control_plane'][0] }}" + run_once: yes + changed_when: false + when: + - drain_nodes + - drain_pod_selector + + - name: Ensure minimum version for drain label selector if necessary + assert: + that: "kubectl_version.stdout.split(' ')[-1] is version('v1.10.0', '>=')" + when: + - drain_nodes + - drain_pod_selector + + - name: Drain node + command: >- + {{ kubectl }} drain + --force + --ignore-daemonsets + --grace-period {{ hostvars['localhost']['drain_grace_period_after_failure'] | default(drain_grace_period) }} + --timeout {{ hostvars['localhost']['drain_timeout_after_failure'] | default(drain_timeout) }} + --delete-emptydir-data {{ kube_override_hostname | default(inventory_hostname) }} + {% if drain_pod_selector %}--pod-selector '{{ drain_pod_selector }}'{% endif %} + when: drain_nodes + register: result + failed_when: + - result.rc != 0 + - not drain_fallback_enabled + until: result.rc == 0 + retries: "{{ drain_retries }}" + delay: "{{ drain_retry_delay_seconds }}" + + - name: Drain fallback + when: + - drain_nodes + - drain_fallback_enabled + - result.rc != 0 + block: + - name: Set facts after regular drain has failed + set_fact: + drain_grace_period_after_failure: "{{ drain_fallback_grace_period }}" + drain_timeout_after_failure: "{{ drain_fallback_timeout }}" + delegate_to: localhost + delegate_facts: yes + run_once: yes + + - name: Drain node - fallback with disabled eviction + command: >- + {{ kubectl }} drain + --force + --ignore-daemonsets + --grace-period {{ drain_fallback_grace_period }} + --timeout {{ drain_fallback_timeout }} + --delete-emptydir-data {{ kube_override_hostname | default(inventory_hostname) }} + {% if drain_pod_selector %}--pod-selector '{{ drain_pod_selector }}'{% endif %} + --disable-eviction + register: drain_fallback_result + until: drain_fallback_result.rc == 0 + retries: "{{ drain_fallback_retries }}" + delay: "{{ drain_fallback_retry_delay_seconds }}" + changed_when: drain_fallback_result.rc == 0 + + rescue: + - name: Set node back to schedulable + command: "{{ kubectl }} uncordon {{ kube_override_hostname | default(inventory_hostname) }}" + when: upgrade_node_uncordon_after_drain_failure + - name: Fail after rescue + fail: + msg: "Failed to drain node {{ kube_override_hostname | default(inventory_hostname) }}" + when: upgrade_node_fail_if_drain_fails diff --git a/kubespray/roles/upgrade/system-upgrade/tasks/apt.yml b/kubespray/roles/upgrade/system-upgrade/tasks/apt.yml new file mode 100644 index 0000000..992bbce --- /dev/null +++ b/kubespray/roles/upgrade/system-upgrade/tasks/apt.yml @@ -0,0 +1,13 @@ +--- +- name: APT Dist-Upgrade + apt: + upgrade: dist + autoremove: true + dpkg_options: force-confold,force-confdef + register: apt_upgrade + +- name: Reboot after APT Dist-Upgrade # noqa no-handler + when: + - apt_upgrade.changed or system_upgrade_reboot == 'always' + - system_upgrade_reboot != 'never' + reboot: diff --git a/kubespray/roles/upgrade/system-upgrade/tasks/main.yml b/kubespray/roles/upgrade/system-upgrade/tasks/main.yml new file mode 100644 index 0000000..61561b1 --- /dev/null +++ b/kubespray/roles/upgrade/system-upgrade/tasks/main.yml @@ -0,0 +1,17 @@ +--- +- name: APT upgrade + when: + - system_upgrade + - ansible_os_family == "Debian" + include_tasks: apt.yml + tags: + - system-upgrade-apt + +- name: YUM upgrade + when: + - system_upgrade + - ansible_os_family == "RedHat" + - not is_fedora_coreos + include_tasks: yum.yml + tags: + - system-upgrade-yum diff --git a/kubespray/roles/upgrade/system-upgrade/tasks/yum.yml b/kubespray/roles/upgrade/system-upgrade/tasks/yum.yml new file mode 100644 index 0000000..6a27177 --- /dev/null +++ b/kubespray/roles/upgrade/system-upgrade/tasks/yum.yml @@ -0,0 +1,12 @@ +--- +- name: YUM upgrade all packages # noqa package-latest + yum: + name: '*' + state: latest + register: yum_upgrade + +- name: Reboot after YUM upgrade # noqa no-handler + when: + - yum_upgrade.changed or system_upgrade_reboot == 'always' + - system_upgrade_reboot != 'never' + reboot: diff --git a/kubespray/roles/win_nodes/kubernetes_patch/defaults/main.yml b/kubespray/roles/win_nodes/kubernetes_patch/defaults/main.yml new file mode 100644 index 0000000..954cb51 --- /dev/null +++ b/kubespray/roles/win_nodes/kubernetes_patch/defaults/main.yml @@ -0,0 +1,4 @@ +--- + +kubernetes_user_manifests_path: "{{ ansible_env.HOME }}/kube-manifests" +kube_proxy_nodeselector: "kubernetes.io/os" diff --git a/kubespray/roles/win_nodes/kubernetes_patch/tasks/main.yml b/kubespray/roles/win_nodes/kubernetes_patch/tasks/main.yml new file mode 100644 index 0000000..880c58c --- /dev/null +++ b/kubespray/roles/win_nodes/kubernetes_patch/tasks/main.yml @@ -0,0 +1,41 @@ +--- + +- name: Ensure that user manifests directory exists + file: + path: "{{ kubernetes_user_manifests_path }}/kubernetes" + state: directory + recurse: yes + tags: [init, cni] + +- name: Apply kube-proxy nodeselector + tags: init + when: + - kube_proxy_deployed + block: + # Due to https://github.com/kubernetes/kubernetes/issues/58212 we cannot rely on exit code for "kubectl patch" + - name: Check current nodeselector for kube-proxy daemonset + command: >- + {{ kubectl }} + get ds kube-proxy --namespace=kube-system + -o jsonpath={.spec.template.spec.nodeSelector.{{ kube_proxy_nodeselector | regex_replace('\.', '\\.') }}} + register: current_kube_proxy_state + retries: 60 + delay: 5 + until: current_kube_proxy_state is succeeded + changed_when: false + + - name: Apply nodeselector patch for kube-proxy daemonset + command: > + {{ kubectl }} + patch ds kube-proxy --namespace=kube-system --type=strategic -p + '{"spec":{"template":{"spec":{"nodeSelector":{"{{ kube_proxy_nodeselector }}":"linux"} }}}}' + register: patch_kube_proxy_state + when: current_kube_proxy_state.stdout | trim | lower != "linux" + + - debug: # noqa name[missing] + msg: "{{ patch_kube_proxy_state.stdout_lines }}" + when: patch_kube_proxy_state is not skipped + + - debug: # noqa name[missing] + msg: "{{ patch_kube_proxy_state.stderr_lines }}" + when: patch_kube_proxy_state is not skipped diff --git a/kubespray/run.rc b/kubespray/run.rc new file mode 100644 index 0000000..570f0dd --- /dev/null +++ b/kubespray/run.rc @@ -0,0 +1,46 @@ +# use virtualenv to install all python requirements +VENVDIR=venv +python3 -m venv $VENVDIR +source $VENVDIR/bin/activate +pip install --upgrade pip +pip install wheel +pip install --upgrade setuptools +pip install -r requirements.txt +pip install -r tests/requirements.txt +ansible-galaxy install -r tests/requirements.yml +pre-commit install +# prepare an inventory to test with +INV=inventory/lab +rm -rf ${INV}.bak &> /dev/null +mv ${INV} ${INV}.bak &> /dev/null +cp -a inventory/sample ${INV} +rm -f ${INV}/hosts.ini + +# customize the vagrant environment +mkdir vagrant +cat << EOF > vagrant/config.rb +\$instance_name_prefix = kub" +\$vm_cpus = 2 +\$num_instances = 3 +\$os = "almalinux8" +\$subnet = "192.168.56" +\$network_plugin = "calico" +\$inventory = "$INV" +\$shared_folders = { 'temp/docker_rpms' => "/var/cache/yum/x86_64/7/docker-ce/packages" } +EOF + +# make the rpm cache +mkdir -p temp/docker_rpms + +vagrant up + +# make a copy of the downloaded docker rpm, to speed up the next provisioning run +scp kub-1:/var/cache/yum/x86_64/7/docker-ce/packages/* temp/docker_rpms/ + +# copy kubectl access configuration in place +mkdir $HOME/.kube/ &> /dev/null +ln -s $PWD/$INV/artifacts/admin.conf $HOME/.kube/config +# make the kubectl binary available +sudo ln -s $PWD/$INV/artifacts/kubectl /usr/local/bin/kubectl +#or +export PATH=$PATH:$PWD/$INV/artifacts diff --git a/kubespray/scale.yml b/kubespray/scale.yml new file mode 100644 index 0000000..b78fc69 --- /dev/null +++ b/kubespray/scale.yml @@ -0,0 +1,3 @@ +--- +- name: Scale the cluster + ansible.builtin.import_playbook: playbooks/scale.yml \ No newline at end of file diff --git a/kubespray/scripts/collect-info.yaml b/kubespray/scripts/collect-info.yaml new file mode 100644 index 0000000..923a6a8 --- /dev/null +++ b/kubespray/scripts/collect-info.yaml @@ -0,0 +1,153 @@ +--- +- name: Collect debug info + hosts: all + become: true + gather_facts: no + + vars: + docker_bin_dir: /usr/bin + bin_dir: /usr/local/bin + ansible_ssh_pipelining: true + etcd_cert_dir: /etc/ssl/etcd/ssl + kube_network_plugin: calico + archive_dirname: collect-info + commands: + - name: timedate_info + cmd: timedatectl status + - name: kernel_info + cmd: uname -r + - name: docker_info + cmd: "{{ docker_bin_dir }}/docker info" + - name: ip_info + cmd: ip -4 -o a + - name: route_info + cmd: ip ro + - name: proc_info + cmd: ps auxf | grep -v ]$ + - name: systemctl_failed_info + cmd: systemctl --state=failed --no-pager + - name: k8s_info + cmd: "{{ bin_dir }}/kubectl get all --all-namespaces -o wide" + - name: errors_info + cmd: journalctl -p err --no-pager + - name: etcd_info + cmd: "{{ bin_dir }}/etcdctl endpoint --cluster health" + - name: calico_info + cmd: "{{ bin_dir }}/calicoctl node status" + when: '{{ kube_network_plugin == "calico" }}' + - name: calico_workload_info + cmd: "{{ bin_dir }}/calicoctl get workloadEndpoint -o wide" + when: '{{ kube_network_plugin == "calico" }}' + - name: calico_pool_info + cmd: "{{ bin_dir }}/calicoctl get ippool -o wide" + when: '{{ kube_network_plugin == "calico" }}' + - name: weave_info + cmd: weave report + when: '{{ kube_network_plugin == "weave" }}' + - name: weave_logs + cmd: "{{ docker_bin_dir }}/docker logs weave" + when: '{{ kube_network_plugin == "weave" }}' + - name: kube_describe_all + cmd: "{{ bin_dir }}/kubectl describe all --all-namespaces" + - name: kube_describe_nodes + cmd: "{{ bin_dir }}/kubectl describe nodes" + - name: kubelet_logs + cmd: journalctl -u kubelet --no-pager + - name: coredns_logs + cmd: "for i in `{{ bin_dir }}/kubectl get pods -n kube-system -l k8s-app=coredns -o jsonpath={.items..metadata.name}`; + do {{ bin_dir }}/kubectl logs ${i} -n kube-system; done" + - name: apiserver_logs + cmd: "for i in `{{ bin_dir }}/kubectl get pods -n kube-system -l component=kube-apiserver -o jsonpath={.items..metadata.name}`; + do {{ bin_dir }}/kubectl logs ${i} -n kube-system; done" + - name: controller_logs + cmd: "for i in `{{ bin_dir }}/kubectl get pods -n kube-system -l component=kube-controller-manager -o jsonpath={.items..metadata.name}`; + do {{ bin_dir }}/kubectl logs ${i} -n kube-system; done" + - name: scheduler_logs + cmd: "for i in `{{ bin_dir }}/kubectl get pods -n kube-system -l component=kube-scheduler -o jsonpath={.items..metadata.name}`; + do {{ bin_dir }}/kubectl logs ${i} -n kube-system; done" + - name: proxy_logs + cmd: "for i in `{{ bin_dir }}/kubectl get pods -n kube-system -l k8s-app=kube-proxy -o jsonpath={.items..metadata.name}`; + do {{ bin_dir }}/kubectl logs ${i} -n kube-system; done" + - name: nginx_logs + cmd: "for i in `{{ bin_dir }}/kubectl get pods -n kube-system -l k8s-app=kube-nginx -o jsonpath={.items..metadata.name}`; + do {{ bin_dir }}/kubectl logs ${i} -n kube-system; done" + - name: flannel_logs + cmd: "for i in `{{ bin_dir }}/kubectl get pods -n kube-system -l app=flannel -o jsonpath={.items..metadata.name}`; + do {{ bin_dir }}/kubectl logs ${i} -n kube-system flannel-container; done" + when: '{{ kube_network_plugin == "flannel" }}' + - name: canal_logs + cmd: "for i in `{{ bin_dir }}/kubectl get pods -n kube-system -l k8s-app=canal-node -o jsonpath={.items..metadata.name}`; + do {{ bin_dir }}/kubectl logs ${i} -n kube-system flannel; done" + when: '{{ kube_network_plugin == "canal" }}' + - name: calico_policy_logs + cmd: "for i in `{{ bin_dir }}/kubectl get pods -n kube-system -l k8s-app=calico-kube-controllers -o jsonpath={.items..metadata.name}`; + do {{ bin_dir }}/kubectl logs ${i} -n kube-system ; done" + when: '{{ kube_network_plugin in ["canal", "calico"] }}' + - name: helm_show_releases_history + cmd: "for i in `{{ bin_dir }}/helm list -q`; do {{ bin_dir }}/helm history ${i} --col-width=0; done" + when: "{{ helm_enabled | default(true) }}" + + logs: + - /var/log/syslog + - /var/log/daemon.log + - /var/log/kern.log + - /var/log/dpkg.log + - /var/log/apt/history.log + - /var/log/yum.log + - /var/log/messages + - /var/log/dmesg + + environment: + ETCDCTL_API: "3" + ETCDCTL_CERT: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}.pem" + ETCDCTL_KEY: "{{ etcd_cert_dir }}/admin-{{ inventory_hostname }}-key.pem" + ETCDCTL_CACERT: "{{ etcd_cert_dir }}/ca.pem" + ETCDCTL_ENDPOINTS: "{{ etcd_access_addresses }}" + + tasks: + - name: Set etcd_access_addresses + set_fact: + etcd_access_addresses: |- + {% for item in groups['etcd'] -%} + https://{{ item }}:2379{% if not loop.last %},{% endif %} + {%- endfor %} + when: "'etcd' in groups" + + - name: Storing commands output + shell: "{{ item.cmd }} &> {{ item.name }}" + failed_when: false + with_items: "{{ commands }}" + when: item.when | default(True) + no_log: True + + - name: Fetch results + fetch: + src: "{{ item.name }}" + dest: "/tmp/{{ archive_dirname }}/commands" + with_items: "{{ commands }}" + when: item.when | default(True) + failed_when: false + + - name: Fetch logs + fetch: + src: "{{ item }}" + dest: "/tmp/{{ archive_dirname }}/logs" + with_items: "{{ logs }}" + failed_when: false + + - name: Pack results and logs + community.general.archive: + path: "/tmp/{{ archive_dirname }}" + dest: "{{ dir | default('.') }}/logs.tar.gz" + remove: true + mode: 0640 + delegate_to: localhost + connection: local + become: false + run_once: true + + - name: Clean up collected command outputs + file: + path: "{{ item.name }}" + state: absent + with_items: "{{ commands }}" diff --git a/kubespray/scripts/download_hash.py b/kubespray/scripts/download_hash.py new file mode 100644 index 0000000..dfd8446 --- /dev/null +++ b/kubespray/scripts/download_hash.py @@ -0,0 +1,65 @@ +#!/usr/bin/env python3 + +# After a new version of Kubernetes has been released, +# run this script to update roles/download/defaults/main/main.yml +# with new hashes. + +import hashlib +import sys + +import requests +from ruamel.yaml import YAML + +MAIN_YML = "../roles/download/defaults/main/main.yml" + +def open_main_yaml(): + yaml = YAML() + yaml.explicit_start = True + yaml.preserve_quotes = True + yaml.width = 4096 + + with open(MAIN_YML, "r") as main_yml: + data = yaml.load(main_yml) + + return data, yaml + + +def download_hash(versions): + architectures = ["arm", "arm64", "amd64", "ppc64le"] + downloads = ["kubelet", "kubectl", "kubeadm"] + + data, yaml = open_main_yaml() + + for download in downloads: + checksum_name = f"{download}_checksums" + for arch in architectures: + for version in versions: + if not version.startswith("v"): + version = f"v{version}" + url = f"https://dl.k8s.io/release/{version}/bin/linux/{arch}/{download}" + download_file = requests.get(url, allow_redirects=True) + download_file.raise_for_status() + sha256sum = hashlib.sha256(download_file.content).hexdigest() + data[checksum_name][arch][version] = sha256sum + + with open(MAIN_YML, "w") as main_yml: + yaml.dump(data, main_yml) + print(f"\n\nUpdated {MAIN_YML}\n") + + +def usage(): + print(f"USAGE:\n {sys.argv[0]} [k8s_version1] [[k8s_version2]....[k8s_versionN]]") + + +def main(argv=None): + if not argv: + argv = sys.argv[1:] + if not argv: + usage() + return 1 + download_hash(argv) + return 0 + + +if __name__ == "__main__": + sys.exit(main()) diff --git a/kubespray/scripts/download_hash.sh b/kubespray/scripts/download_hash.sh new file mode 100755 index 0000000..b37e466 --- /dev/null +++ b/kubespray/scripts/download_hash.sh @@ -0,0 +1,259 @@ +#!/bin/bash + +set -o errexit +set -o pipefail +if [[ ${DEBUG:-false} == "true" ]]; then + set -o xtrace +fi + +checksums_file="$(git rev-parse --show-toplevel)/roles/download/defaults/main/checksums.yml" +downloads_folder=/tmp/kubespray_binaries + +function get_versions { + local type="$1" + local name="$2" + # NOTE: Limit in the number of versions to be register in the checksums file + local limit="${3:-7}" + local python_app="${4:-"import sys,re;tags=[tag.rstrip() for tag in sys.stdin if re.match(\'^v?(0|[1-9]\\d*)\\.(0|[1-9]\\d*)\\.(0|[1-9]\\d*)$\',tag)];print(\' \'.join(tags[:$limit]))"}" + local version="" + local attempt_counter=0 + readonly max_attempts=5 + + until [ "$version" ]; do + version=$("_get_$type" "$name" "$python_app") + if [ "$version" ]; then + break + elif [ ${attempt_counter} -eq ${max_attempts} ]; then + echo "Max attempts reached" + exit 1 + fi + attempt_counter=$((attempt_counter + 1)) + sleep $((attempt_counter * 2)) + done + + echo "${version}" +} + +function _get_github_tags { + local repo="$1" + local python_app="$2" + + # The number of results per page (max 100). + tags="$(curl -s "https://api.github.com/repos/$repo/tags?per_page=100")" + if [ "$tags" ]; then + echo "$tags" | grep -Po '"name":.*?[^\\]",' | awk -F '"' '{print $4}' | python -c "$python_app" + fi +} + +function _vercmp { + local v1=$1 + local op=$2 + local v2=$3 + local result + + # sort the two numbers with sort's "-V" argument. Based on if v2 + # swapped places with v1, we can determine ordering. + result=$(echo -e "$v1\n$v2" | sort -V | head -1) + + case $op in + "==") + [ "$v1" = "$v2" ] + return + ;; + ">") + [ "$v1" != "$v2" ] && [ "$result" = "$v2" ] + return + ;; + "<") + [ "$v1" != "$v2" ] && [ "$result" = "$v1" ] + return + ;; + ">=") + [ "$result" = "$v2" ] + return + ;; + "<=") + [ "$result" = "$v1" ] + return + ;; + *) + echo "unrecognised op: $op" + exit 1 + ;; + esac +} + +function get_checksums { + local binary="$1" + local version_exceptions="cri_dockerd_archive nerdctl_archive containerd_archive youki" + declare -A skip_archs=( +["crio_archive"]="arm ppc64le" +["calicoctl_binary"]="arm" +["ciliumcli_binary"]="ppc64le" +["etcd_binary"]="arm" +["cri_dockerd_archive"]="arm ppc64le" +["runc"]="arm" +["crun"]="arm ppc64le" +["youki"]="arm arm64 ppc64le" +["kata_containers_binary"]="arm arm64 ppc64le" +["gvisor_runsc_binary"]="arm ppc64le" +["gvisor_containerd_shim_binary"]="arm ppc64le" +["containerd_archive"]="arm" +["skopeo_binary"]="arm ppc64le" +) + echo "${binary}_checksums:" | tee --append "$checksums_file" + for arch in arm arm64 amd64 ppc64le; do + echo " $arch:" | tee --append "$checksums_file" + for version in "${@:2}"; do + checksum=0 + [[ "${skip_archs[$binary]}" == *"$arch"* ]] || checksum=$(_get_checksum "$binary" "$version" "$arch") + [[ "$version_exceptions" != *"$binary"* ]] || version=${version#v} + echo " $version: $checksum" | tee --append "$checksums_file" + done + done +} + +function get_krew_archive_checksums { + declare -A archs=( +["linux"]="arm arm64 amd64" +["darwin"]="arm64 amd64" +["windows"]="amd64" +) + + echo "krew_archive_checksums:" | tee --append "$checksums_file" + for os in "${!archs[@]}"; do + echo " $os:" | tee --append "$checksums_file" + for arch in arm arm64 amd64 ppc64le; do + echo " $arch:" | tee --append "$checksums_file" + for version in "$@"; do + checksum=0 + [[ " ${archs[$os]} " != *" $arch "* ]] || checksum=$(_get_checksum "krew_archive" "$version" "$arch" "$os") + echo " $version: $checksum" | tee --append "$checksums_file" + done + done + done +} + +function get_calico_crds_archive_checksums { + echo "calico_crds_archive_checksums:" | tee --append "$checksums_file" + for version in "$@"; do + echo " $version: $(_get_checksum "calico_crds_archive" "$version")" | tee --append "$checksums_file" + done +} + +function get_containerd_archive_checksums { + declare -A support_version_history=( +["arm"]="2" +["arm64"]="1.6.0" +["amd64"]="1.5.5" +["ppc64le"]="1.6.7" +) + + echo "containerd_archive_checksums:" | tee --append "$checksums_file" + for arch in arm arm64 amd64 ppc64le; do + echo " $arch:" | tee --append "$checksums_file" + for version in "${@}"; do + _vercmp "${version#v}" '>=' "${support_version_history[$arch]}" && checksum=$(_get_checksum "containerd_archive" "$version" "$arch") || checksum=0 + echo " ${version#v}: $checksum" | tee --append "$checksums_file" + done + done +} + +function get_k8s_checksums { + local binary=$1 + + echo "${binary}_checksums:" | tee --append "$checksums_file" + echo " arm:" | tee --append "$checksums_file" + for version in "${@:2}"; do + _vercmp "${version#v}" '<' "1.27" && checksum=$(_get_checksum "$binary" "$version" "arm") || checksum=0 + echo " ${version}: $checksum" | tee --append "$checksums_file" + done + for arch in arm64 amd64 ppc64le; do + echo " $arch:" | tee --append "$checksums_file" + for version in "${@:2}"; do + echo " ${version}: $(_get_checksum "$binary" "$version" "$arch")" | tee --append "$checksums_file" + done + done +} + +function _get_checksum { + local binary="$1" + local version="$2" + local arch="${3:-amd64}" + local os="${4:-linux}" + local target="$downloads_folder/$binary/$version-$os-$arch" + readonly github_url="https://github.com" + readonly github_releases_url="$github_url/%s/releases/download/$version/%s" + readonly github_archive_url="$github_url/%s/archive/%s" + readonly google_url="https://storage.googleapis.com" + readonly release_url="https://dl.k8s.io" + readonly k8s_url="$release_url/release/$version/bin/$os/$arch/%s" + + # Download URLs + declare -A urls=( +["crictl"]="$(printf "$github_releases_url" "kubernetes-sigs/cri-tools" "crictl-$version-$os-$arch.tar.gz")" +["crio_archive"]="$google_url/cri-o/artifacts/cri-o.$arch.$version.tar.gz" +["kubelet"]="$(printf "$k8s_url" "kubelet")" +["kubectl"]="$(printf "$k8s_url" "kubectl")" +["kubeadm"]="$(printf "$k8s_url" "kubeadm")" +["etcd_binary"]="$(printf "$github_releases_url" "etcd-io/etcd" "etcd-$version-$os-$arch.tar.gz")" +["cni_binary"]="$(printf "$github_releases_url" "containernetworking/plugins" "cni-plugins-$os-$arch-$version.tgz")" +["calicoctl_binary"]="$(printf "$github_releases_url" "projectcalico/calico" "calicoctl-$os-$arch")" +["ciliumcli_binary"]="$(printf "$github_releases_url" "cilium/cilium-cli" "cilium-$os-$arch.tar.gz")" +["calico_crds_archive"]="$(printf "$github_archive_url" "projectcalico/calico" "$version.tar.gz")" +["krew_archive"]="$(printf "$github_releases_url" "kubernetes-sigs/krew" "krew-${os}_$arch.tar.gz")" +["helm_archive"]="https://get.helm.sh/helm-$version-$os-$arch.tar.gz" +["cri_dockerd_archive"]="$(printf "$github_releases_url" "Mirantis/cri-dockerd" "cri-dockerd-${version#v}.$arch.tgz")" +["runc"]="$(printf "$github_releases_url" "opencontainers/runc" "runc.$arch")" +["crun"]="$(printf "$github_releases_url" "containers/crun" "crun-$version-$os-$arch")" +["youki"]="$(printf "$github_releases_url" "containers/youki" "youki_$([ $version == "v0.0.1" ] && echo "v0_0_1" || echo "${version#v}" | sed 's|\.|_|g')_$os.tar.gz")" +["kata_containers_binary"]="$(printf "$github_releases_url" "kata-containers/kata-containers" "kata-static-$version-${arch//amd64/x86_64}.tar.xz")" +["gvisor_runsc_binary"]="$(printf "$google_url/gvisor/releases/release/$version/%s/runsc" "$(echo "$arch" | sed -e 's/amd64/x86_64/' -e 's/arm64/aarch64/')")" +["gvisor_containerd_shim_binary"]="$(printf "$google_url/gvisor/releases/release/$version/%s/containerd-shim-runsc-v1" "$(echo "$arch" | sed -e 's/amd64/x86_64/' -e 's/arm64/aarch64/')")" +["nerdctl_archive"]="$(printf "$github_releases_url" "containerd/nerdctl" "nerdctl-${version#v}-$os-$([ "$arch" == "arm" ] && echo "arm-v7" || echo "$arch" ).tar.gz")" +["containerd_archive"]="$(printf "$github_releases_url" "containerd/containerd" "containerd-${version#v}-$os-$arch.tar.gz")" +["skopeo_binary"]="$(printf "$github_releases_url" "lework/skopeo-binary" "skopeo-$os-$arch")" +["yq"]="$(printf "$github_releases_url" "mikefarah/yq" "yq_${os}_$arch")" +) + + mkdir -p "$(dirname $target)" + [ -f "$target" ] || curl -LfSs -o "${target}" "${urls[$binary]}" + sha256sum ${target} | awk '{print $1}' +} + +function main { + mkdir -p "$(dirname "$checksums_file")" + echo "---" | tee "$checksums_file" + get_checksums crictl $(get_versions github_tags kubernetes-sigs/cri-tools 4) + get_checksums crio_archive $(get_versions github_tags cri-o/cri-o) + kubernetes_versions=$(get_versions github_tags kubernetes/kubernetes 25) + echo "# Checksum" | tee --append "$checksums_file" + echo "# Kubernetes versions above Kubespray's current target version are untested and should be used with caution." | tee --append "$checksums_file" + get_k8s_checksums kubelet $kubernetes_versions + get_checksums kubectl $kubernetes_versions + get_k8s_checksums kubeadm $kubernetes_versions + get_checksums etcd_binary $(get_versions github_tags etcd-io/etcd) + get_checksums cni_binary $(get_versions github_tags containernetworking/plugins) + calico_versions=$(get_versions github_tags projectcalico/calico 20) + get_checksums calicoctl_binary $calico_versions + get_checksums ciliumcli_binary $(get_versions github_tags cilium/cilium-cli 10) + get_calico_crds_archive_checksums $calico_versions + get_krew_archive_checksums $(get_versions github_tags kubernetes-sigs/krew 2) + get_checksums helm_archive $(get_versions github_tags helm/helm) + get_checksums cri_dockerd_archive $(get_versions github_tags Mirantis/cri-dockerd) + get_checksums runc $(get_versions github_tags opencontainers/runc 5) + get_checksums crun $(get_versions github_tags containers/crun) + get_checksums youki $(get_versions github_tags containers/youki) + get_checksums kata_containers_binary $(get_versions github_tags kata-containers/kata-containers 10) + gvisor_versions=$(get_versions github_tags google/gvisor 0 "import sys,re;tags=[tag[8:16] for tag in sys.stdin if re.match('^release-?(0|[1-9]\d*)\.(0|[1-9]\d*)$',tag)];print(' '.join(tags[:9]))") + get_checksums gvisor_runsc_binary $gvisor_versions + get_checksums gvisor_containerd_shim_binary $gvisor_versions + get_checksums nerdctl_archive $(get_versions github_tags containerd/nerdctl) + get_containerd_archive_checksums $(get_versions github_tags containerd/containerd 30) + get_checksums skopeo_binary $(get_versions github_tags lework/skopeo-binary) + get_checksums yq $(get_versions github_tags mikefarah/yq) +} + +if [[ ${__name__:-"__main__"} == "__main__" ]]; then + main +fi diff --git a/kubespray/scripts/gen_tags.sh b/kubespray/scripts/gen_tags.sh new file mode 100755 index 0000000..1bc94c8 --- /dev/null +++ b/kubespray/scripts/gen_tags.sh @@ -0,0 +1,12 @@ +#!/bin/sh +set -eo pipefail + +#Generate MD formatted tags from roles and cluster yaml files +printf "|%25s |%9s\n" "Tag name" "Used for" +echo "|--------------------------|---------" +tags=$(grep -r tags: . | perl -ne '/tags:\s\[?(([\w\-_]+,?\s?)+)/ && printf "%s ", "$1"'|\ + perl -ne 'print join "\n", split /\s|,/' | sort -u) +for tag in $tags; do + match=$(cat docs/ansible.md | perl -ne "/^\|\s+${tag}\s\|\s+((\S+\s?)+)/ && printf \$1") + printf "|%25s |%s\n" "${tag}" " ${match}" +done diff --git a/kubespray/scripts/gitlab-branch-cleanup/.gitignore b/kubespray/scripts/gitlab-branch-cleanup/.gitignore new file mode 100644 index 0000000..03e7ca8 --- /dev/null +++ b/kubespray/scripts/gitlab-branch-cleanup/.gitignore @@ -0,0 +1,2 @@ +openrc +venv diff --git a/kubespray/scripts/gitlab-branch-cleanup/README.md b/kubespray/scripts/gitlab-branch-cleanup/README.md new file mode 100644 index 0000000..6a2b5ff --- /dev/null +++ b/kubespray/scripts/gitlab-branch-cleanup/README.md @@ -0,0 +1,24 @@ +# gitlab-branch-cleanup + +Cleanup old branches in a GitLab project + +## Installation + +```shell +pip install -r requirements.txt +python main.py --help +``` + +## Usage + +```console +$ export GITLAB_API_TOKEN=foobar +$ python main.py kargo-ci/kubernetes-sigs-kubespray +Deleting branch pr-5220-containerd-systemd from 2020-02-17 ... +Deleting branch pr-5561-feature/cinder_csi_fixes from 2020-02-17 ... +Deleting branch pr-5607-add-flatcar from 2020-02-17 ... +Deleting branch pr-5616-fix-typo from 2020-02-17 ... +Deleting branch pr-5634-helm_310 from 2020-02-18 ... +Deleting branch pr-5644-patch-1 from 2020-02-15 ... +Deleting branch pr-5647-master from 2020-02-17 ... +``` diff --git a/kubespray/scripts/gitlab-branch-cleanup/main.py b/kubespray/scripts/gitlab-branch-cleanup/main.py new file mode 100644 index 0000000..2d7fe1c --- /dev/null +++ b/kubespray/scripts/gitlab-branch-cleanup/main.py @@ -0,0 +1,38 @@ +import gitlab +import argparse +import os +import sys +from datetime import timedelta, datetime, timezone + + +parser = argparse.ArgumentParser( + description='Cleanup old branches in a GitLab project') +parser.add_argument('--api', default='https://gitlab.com/', + help='URL of GitLab API, defaults to gitlab.com') +parser.add_argument('--age', type=int, default=30, + help='Delete branches older than this many days') +parser.add_argument('--prefix', default='pr-', + help='Cleanup only branches with names matching this prefix') +parser.add_argument('--dry-run', action='store_true', + help='Do not delete anything') +parser.add_argument('project', + help='Path of the GitLab project') + +args = parser.parse_args() +limit = datetime.now(timezone.utc) - timedelta(days=args.age) + +if os.getenv('GITLAB_API_TOKEN', '') == '': + print("Environment variable GITLAB_API_TOKEN is required.") + sys.exit(2) + +gl = gitlab.Gitlab(args.api, private_token=os.getenv('GITLAB_API_TOKEN')) +gl.auth() + +p = gl.projects.get(args.project) +for b in p.branches.list(all=True): + date = datetime.fromisoformat(b.commit['created_at']) + if date < limit and not b.protected and not b.default and b.name.startswith(args.prefix): + print("Deleting branch %s from %s ..." % + (b.name, date.date().isoformat())) + if not args.dry_run: + b.delete() diff --git a/kubespray/scripts/gitlab-branch-cleanup/requirements.txt b/kubespray/scripts/gitlab-branch-cleanup/requirements.txt new file mode 100644 index 0000000..4a169ed --- /dev/null +++ b/kubespray/scripts/gitlab-branch-cleanup/requirements.txt @@ -0,0 +1 @@ +python-gitlab diff --git a/kubespray/scripts/gitlab-runner.sh b/kubespray/scripts/gitlab-runner.sh new file mode 100644 index 0000000..c05ee7e --- /dev/null +++ b/kubespray/scripts/gitlab-runner.sh @@ -0,0 +1,22 @@ +#!/bin/sh + +docker run -d --name gitlab-runner --restart always -v /srv/gitlab-runner/cache:/srv/gitlab-runner/cache -v /srv/gitlab-runner/config:/etc/gitlab-runner -v /var/run/docker.sock:/var/run/docker.sock gitlab/gitlab-runner:v1.10.0 + +# +#/srv/gitlab-runner/config# cat config.toml +#concurrent = 10 +#check_interval = 1 + +#[[runners]] +# name = "2edf3d71fe19" +# url = "https://gitlab.com" +# token = "THE TOKEN-CHANGEME" +# executor = "docker" +# [runners.docker] +# tls_verify = false +# image = "docker:latest" +# privileged = true +# disable_cache = false +# cache_dir = "/srv/gitlab-runner/cache" +# volumes = ["/var/run/docker.sock:/var/run/docker.sock", "/srv/gitlab-runner/cache:/cache:rw"] +# [runners.cache] diff --git a/kubespray/scripts/openstack-cleanup/.gitignore b/kubespray/scripts/openstack-cleanup/.gitignore new file mode 100644 index 0000000..61f5948 --- /dev/null +++ b/kubespray/scripts/openstack-cleanup/.gitignore @@ -0,0 +1 @@ +openrc diff --git a/kubespray/scripts/openstack-cleanup/README.md b/kubespray/scripts/openstack-cleanup/README.md new file mode 100644 index 0000000..737d2f6 --- /dev/null +++ b/kubespray/scripts/openstack-cleanup/README.md @@ -0,0 +1,21 @@ +# openstack-cleanup + +Tool to deletes openstack servers older than a specific age (default 4h). + +Useful to cleanup orphan servers that are left behind when CI is manually cancelled or fails unexpectedly. + +## Installation + +```shell +pip install -r requirements.txt +python main.py --help +``` + +## Usage + +```console +$ python main.py +This will delete VMs... (ctrl+c to cancel) +Will delete server example1 +Will delete server example2 +``` diff --git a/kubespray/scripts/openstack-cleanup/main.py b/kubespray/scripts/openstack-cleanup/main.py new file mode 100755 index 0000000..2ddccc0 --- /dev/null +++ b/kubespray/scripts/openstack-cleanup/main.py @@ -0,0 +1,98 @@ +#!/usr/bin/env python +import argparse +import openstack +import logging +import datetime +import time + +DATE_FORMAT = '%Y-%m-%dT%H:%M:%SZ' +PAUSE_SECONDS = 5 + +log = logging.getLogger('openstack-cleanup') + +parser = argparse.ArgumentParser(description='Cleanup OpenStack resources') + +parser.add_argument('-v', '--verbose', action='store_true', + help='Increase verbosity') +parser.add_argument('--hours', type=int, default=4, + help='Age (in hours) of VMs to cleanup (default: 4h)') +parser.add_argument('--dry-run', action='store_true', + help='Do not delete anything') + +args = parser.parse_args() + +oldest_allowed = datetime.datetime.now() - datetime.timedelta(hours=args.hours) + + +def main(): + if args.dry_run: + print('Running in dry-run mode') + else: + print('This will delete resources... (ctrl+c to cancel)') + time.sleep(PAUSE_SECONDS) + + conn = openstack.connect() + + print('Servers...') + map_if_old(conn.compute.delete_server, + conn.compute.servers()) + + print('Security groups...') + map_if_old(conn.network.delete_security_group, + conn.network.security_groups()) + + print('Ports...') + try: + map_if_old(conn.network.delete_port, + conn.network.ports()) + except openstack.exceptions.ConflictException as ex: + # Need to find subnet-id which should be removed from a router + for sn in conn.network.subnets(): + try: + fn_if_old(conn.network.delete_subnet, sn) + except openstack.exceptions.ConflictException: + for r in conn.network.routers(): + print("Deleting subnet %s from router %s", sn, r) + try: + conn.network.remove_interface_from_router( + r, subnet_id=sn.id) + except Exception as ex: + print("Failed to delete subnet from router as %s", ex) + + for ip in conn.network.ips(): + fn_if_old(conn.network.delete_ip, ip) + + # After removing unnecessary subnet from router, retry to delete ports + map_if_old(conn.network.delete_port, + conn.network.ports()) + + print('Subnets...') + map_if_old(conn.network.delete_subnet, + conn.network.subnets()) + + print('Networks...') + for n in conn.network.networks(): + if not n.is_router_external: + fn_if_old(conn.network.delete_network, n) + + +# runs the given fn to all elements of the that are older than allowed +def map_if_old(fn, items): + for item in items: + fn_if_old(fn, item) + + +# run the given fn function only if the passed item is older than allowed +def fn_if_old(fn, item): + created_at = datetime.datetime.strptime(item.created_at, DATE_FORMAT) + if item.name == "default": # skip default security group + return + if created_at < oldest_allowed: + print('Will delete %(name)s (%(id)s)' % item) + if not args.dry_run: + fn(item) + + +if __name__ == '__main__': + # execute only if run as a script + main() diff --git a/kubespray/scripts/openstack-cleanup/requirements.txt b/kubespray/scripts/openstack-cleanup/requirements.txt new file mode 100644 index 0000000..426c1b0 --- /dev/null +++ b/kubespray/scripts/openstack-cleanup/requirements.txt @@ -0,0 +1,2 @@ +openstacksdk>=0.43.0 +six diff --git a/kubespray/scripts/premoderator.sh b/kubespray/scripts/premoderator.sh new file mode 100644 index 0000000..ab1a7ef --- /dev/null +++ b/kubespray/scripts/premoderator.sh @@ -0,0 +1,51 @@ +#!/bin/bash +# A naive premoderation script to allow Gitlab CI pipeline on a specific PRs' comment +# Exits with 0, if the pipeline is good to go +# Exits with 1, if the user is not allowed to start pipeline +# Exits with 2, if script is unable to get issue id from CI_COMMIT_REF_NAME variable +# Exits with 3, if missing the magic comment in the pipeline to start the pipeline + +CURL_ARGS="-fs --retry 4 --retry-delay 5" +MAGIC="${MAGIC:-ci check this}" +exit_code=0 + +# Get PR number from CI_COMMIT_REF_NAME +issue=$(echo ${CI_COMMIT_REF_NAME} | perl -ne '/^pr-(\d+)-\S+$/ && print $1') + +if [ "$issue" = "" ]; then + echo "Unable to get issue id from: $CI_COMMIT_REF_NAME" + exit 2 +fi + +echo "Fetching labels from PR $issue" +labels=$(curl ${CURL_ARGS} "https://api.github.com/repos/kubernetes-sigs/kubespray/issues/${issue}?access_token=${GITHUB_TOKEN}" | jq '{labels: .labels}' | jq '.labels[].name' | jq -s '') +labels_to_patch=$(echo -n $labels | jq '. + ["needs-ci-auth"]' | tr -d "\n") + +echo "Checking for '$MAGIC' comment in PR $issue" + +# Get the user name from the PR comments with the wanted magic incantation casted +user=$(curl ${CURL_ARGS} "https://api.github.com/repos/kubernetes-sigs/kubespray/issues/${issue}/comments" | jq -M "map(select(.body | contains (\"$MAGIC\"))) | .[0] .user.login" | tr -d '"') + +# Check for the required user group membership to allow (exit 0) or decline (exit >0) the pipeline +if [ "$user" = "" ] || [ "$user" = "null" ]; then + echo "Missing '$MAGIC' comment from one of the OWNERS" + exit_code=3 +else + echo "Found comment from user: $user" + + curl ${CURL_ARGS} "https://api.github.com/orgs/kubernetes-sigs/members/${user}" + + if [ $? -ne 0 ]; then + echo "User does not have permissions to start CI run" + exit_code=1 + else + labels_to_patch=$(echo -n $labels | jq '. - ["needs-ci-auth"]' | tr -d "\n") + exit_code=0 + echo "$user has allowed CI to start" + fi +fi + +# Patch labels on PR +curl ${CURL_ARGS} --request PATCH "https://api.github.com/repos/kubernetes-sigs/kubespray/issues/${issue}?access_token=${GITHUB_TOKEN}" -H "Content-Type: application/json" -d "{\"labels\": ${labels_to_patch}}" + +exit $exit_code diff --git a/kubespray/setup.cfg b/kubespray/setup.cfg new file mode 100644 index 0000000..96f50b6 --- /dev/null +++ b/kubespray/setup.cfg @@ -0,0 +1,62 @@ +[metadata] +name = kubespray +summary = Ansible modules for installing Kubernetes +description-file = + README.md +author = Kubespray +author-email = smainklh@gmail.com +license = Apache License (2.0) +home-page = https://github.com/kubernetes-sigs/kubespray +classifier = + License :: OSI Approved :: Apache Software License + Development Status :: 4 - Beta + Intended Audience :: Developers + Intended Audience :: System Administrators + Intended Audience :: Information Technology + Topic :: Utilities + +[global] +setup-hooks = + pbr.hooks.setup_hook + +[files] +data_files = + usr/share/kubespray/playbooks/ = + cluster.yml + upgrade-cluster.yml + scale.yml + reset.yml + remove-node.yml + extra_playbooks/upgrade-only-k8s.yml + usr/share/kubespray/roles = roles/* + usr/share/kubespray/library = library/* + usr/share/doc/kubespray/ = + LICENSE + README.md + usr/share/doc/kubespray/inventory/ = + inventory/sample/inventory.ini + etc/kubespray/ = + ansible.cfg + etc/kubespray/inventory/sample/group_vars/ = + inventory/sample/group_vars/etcd.yml + etc/kubespray/inventory/sample/group_vars/all/ = + inventory/sample/group_vars/all/all.yml + inventory/sample/group_vars/all/azure.yml + inventory/sample/group_vars/all/coreos.yml + inventory/sample/group_vars/all/docker.yml + inventory/sample/group_vars/all/oci.yml + inventory/sample/group_vars/all/openstack.yml + +[wheel] +universal = 1 + +[pbr] +skip_authors = True +skip_changelog = True + +[bdist_rpm] +group = "System Environment/Libraries" +requires = + ansible + python-jinja2 + python-netaddr diff --git a/kubespray/setup.py b/kubespray/setup.py new file mode 100644 index 0000000..6a931a6 --- /dev/null +++ b/kubespray/setup.py @@ -0,0 +1,19 @@ +# Copyright Red Hat, Inc. All Rights Reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +import setuptools + +setuptools.setup( + setup_requires=['pbr'], + pbr=True) diff --git a/kubespray/test-infra/image-builder/Makefile b/kubespray/test-infra/image-builder/Makefile new file mode 100644 index 0000000..82dba64 --- /dev/null +++ b/kubespray/test-infra/image-builder/Makefile @@ -0,0 +1,2 @@ +deploy: + ansible-playbook -i hosts.ini -e docker_password=$(docker_password) cluster.yml diff --git a/kubespray/test-infra/image-builder/OWNERS b/kubespray/test-infra/image-builder/OWNERS new file mode 100644 index 0000000..0d2e92d --- /dev/null +++ b/kubespray/test-infra/image-builder/OWNERS @@ -0,0 +1,8 @@ +# See the OWNERS docs at https://go.k8s.io/owners + +approvers: + - woopstar + - ant31 +reviewers: + - woopstar + - ant31 diff --git a/kubespray/test-infra/image-builder/cluster.yml b/kubespray/test-infra/image-builder/cluster.yml new file mode 100644 index 0000000..4a622ca --- /dev/null +++ b/kubespray/test-infra/image-builder/cluster.yml @@ -0,0 +1,6 @@ +--- +- name: Build kubevirt images + hosts: image-builder + gather_facts: false + roles: + - kubevirt-images diff --git a/kubespray/test-infra/image-builder/hosts.ini b/kubespray/test-infra/image-builder/hosts.ini new file mode 100644 index 0000000..e000302 --- /dev/null +++ b/kubespray/test-infra/image-builder/hosts.ini @@ -0,0 +1,4 @@ +image-builder-1 ansible_ssh_host=xxx.xxx.xxx.xxx + +[image-builder] +image-builder-1 diff --git a/kubespray/test-infra/image-builder/roles/kubevirt-images/defaults/main.yml b/kubespray/test-infra/image-builder/roles/kubevirt-images/defaults/main.yml new file mode 100644 index 0000000..47d9bee --- /dev/null +++ b/kubespray/test-infra/image-builder/roles/kubevirt-images/defaults/main.yml @@ -0,0 +1,112 @@ +--- +images_dir: /images/base + +docker_user: kubespray+buildvmimages +docker_host: quay.io +registry: quay.io/kubespray + +images: + ubuntu-2004: + filename: focal-server-cloudimg-amd64.img + url: https://cloud-images.ubuntu.com/focal/current/focal-server-cloudimg-amd64-disk-kvm.img + checksum: sha256:8faf1f5a27c956ad0c49dac3114a355fbaf1b2d21709e10a18e67213fbb95b81 + converted: false + tag: "latest" + + ubuntu-2204: + filename: jammy-server-cloudimg-amd64.img + url: https://cloud-images.ubuntu.com/jammy/current/jammy-server-cloudimg-amd64-disk-kvm.img + checksum: sha256:d3f3f446bf35b2e58b82c10c8fa65525264efe5b0e398238f00ab670f49528ab + converted: false + tag: "latest" + + fedora-37: + filename: Fedora-Cloud-Base-37-1.7.x86_64.qcow2 + url: https://download.fedoraproject.org/pub/fedora/linux/releases/37/Cloud/x86_64/images/Fedora-Cloud-Base-37-1.7.x86_64.qcow2 + checksum: sha256:b5b9bec91eee65489a5745f6ee620573b23337cbb1eb4501ce200b157a01f3a0 + converted: true + tag: "latest" + + fedora-38: + filename: Fedora-Cloud-Base-38-1.6.x86_64.qcow2 + url: https://download.fedoraproject.org/pub/fedora/linux/releases/38/Cloud/x86_64/images/Fedora-Cloud-Base-38-1.6.x86_64.qcow2 + checksum: sha256:d334670401ff3d5b4129fcc662cf64f5a6e568228af59076cc449a4945318482 + converted: true + tag: "latest" + + fedora-coreos: + filename: fedora-coreos-32.20200601.3.0-openstack.x86_64.qcow2.xz + url: https://builds.coreos.fedoraproject.org/prod/streams/stable/builds/32.20200601.3.0/x86_64/fedora-coreos-32.20200601.3.0-openstack.x86_64.qcow2.xz + checksum: sha256:fe78c348189d745eb5f6f80ff9eb2af67da8e84880d264f4301faaf7c2a72646 + converted: true + tag: "latest" + + centos-7: + filename: CentOS-7-x86_64-GenericCloud-2009.qcow2 + url: http://cloud.centos.org/centos/7/images/CentOS-7-x86_64-GenericCloud-2211.qcow2 + checksum: sha256:284aab2b23d91318f169ff464bce4d53404a15a0618ceb34562838c59af4adea + converted: true + tag: "latest" + + centos-8: + filename: CentOS-8-GenericCloud-8.3.2011-20201204.2.x86_64.qcow2 + url: http://cloud.centos.org/centos/8/x86_64/images/CentOS-8-GenericCloud-8.3.2011-20201204.2.x86_64.qcow2 + checksum: sha256:7ec97062618dc0a7ebf211864abf63629da1f325578868579ee70c495bed3ba0 + converted: true + tag: "latest" + + almalinux-8: + filename: AlmaLinux-8-GenericCloud-latest.x86_64.qcow2 + url: https://repo.almalinux.org/almalinux/8.5/cloud/x86_64/images/AlmaLinux-8-GenericCloud-8.5-20211119.x86_64.qcow2 + checksum: sha256:d629247b12802157be127db53a7fcb484b80fceae9896d750c953a51a8c6688f + converted: true + tag: "latest" + + rockylinux-8: + filename: Rocky-8-GenericCloud-8.6-20220515.x86_64.qcow2 + url: https://download.rockylinux.org/pub/rocky/8.6/images/Rocky-8-GenericCloud-8.6-20220515.x86_64.qcow2 + checksum: sha256:77e79f487c70f6bfa5655d8084e02cb8d31900a2c2a22b2334c3401b40a1231c + converted: true + tag: "latest" + + rockylinux-9: + filename: Rocky-9-GenericCloud-9.0-20220830.0.x86_64.qcow2 + url: https://download.rockylinux.org/pub/rocky/9.0/images/x86_64/Rocky-9-GenericCloud-9.0-20220830.0.x86_64.qcow2 + checksum: sha256:f02570e0ad3653df7f56baa8157739dbe92a003234acd5824dcf94d24694e20b + converted: true + tag: "latest" + + debian-10: + filename: debian-10-openstack-amd64.qcow2 + url: https://cdimage.debian.org/cdimage/openstack/current-10/debian-10-openstack-amd64.qcow2 + checksum: sha512:296ad8345cb49e52464a0cb8bf4365eb0b9e4220c47ebdd73d134d51effc756d5554aee15027fffd038fef4ad5fa984c94208bce60572d58b2ab26f74bb2a5de + converted: true + tag: "latest" + + debian-11: + filename: debian-11-generic-amd64-20210814-734.qcow2 + url: https://cdimage.debian.org/cdimage/cloud/bullseye/20210814-734/debian-11-generic-amd64-20210814-734.qcow2 + checksum: sha512:ed680265ce925e3e02336b052bb476883e2d3b023f7b7d39d064d58ba5f1856869f75dca637c27c0303b731d082ff23a7e45ea2e3f9bcb8f3c4ce0c24332885d + converted: true + tag: "latest" + + debian-12: + filename: debian-12-generic-amd64-20230612-1409.qcow2 + url: https://cdimage.debian.org/cdimage/cloud/bookworm/20230612-1409/debian-12-generic-amd64-20230612-1409.qcow2 + checksum: sha512:61358292dbec302446a272d5011019091ca78e3fe8878b2d67d31b32e0661306c53a72f793f109394daf937a3db7b2db34422d504e07fdbb300a7bf87109fcf1 + converted: true + tag: "latest" + + oracle-7: + filename: oracle-linux-76.qcow2 + url: https://storage.googleapis.com/born-images/oracle76/oracle-linux-76.qcow2 + checksum: sha256:f396c03e907fa2a0c94d6807b9f62622f23ee3499df4456ae2a15da381fbdca5 + converted: true + tag: "latest" + + opensuse-leap-15: + filename: openSUSE-Leap-15.3.x86_64-1.0.1-NoCloud-Build2.63.qcow2 + url: https://download.opensuse.org/repositories/Cloud:/Images:/Leap_15.3/images/openSUSE-Leap-15.3.x86_64-1.0.1-NoCloud-Build2.63.qcow2 + checksum: sha256:289248945e2d058551c71c1bdcb31a361cefe7136c7fd88a09b524eedfaf5215 + converted: true + tag: "latest" diff --git a/kubespray/test-infra/image-builder/roles/kubevirt-images/tasks/main.yml b/kubespray/test-infra/image-builder/roles/kubevirt-images/tasks/main.yml new file mode 100644 index 0000000..99c1c1c --- /dev/null +++ b/kubespray/test-infra/image-builder/roles/kubevirt-images/tasks/main.yml @@ -0,0 +1,58 @@ +--- + +- name: Create image directory + file: + state: directory + path: "{{ images_dir }}" + mode: 0755 + +- name: Download images files + get_url: + url: "{{ item.value.url }}" + dest: "{{ images_dir }}/{{ item.value.filename }}" + checksum: "{{ item.value.checksum }}" + mode: 0644 + loop: "{{ images | dict2items }}" + +- name: Unxz compressed images + command: unxz --force {{ images_dir }}/{{ item.value.filename }} + loop: "{{ images | dict2items }}" + when: + - item.value.filename.endswith('.xz') + +- name: Convert images which is not in qcow2 format + command: qemu-img convert -O qcow2 {{ images_dir }}/{{ item.value.filename.rstrip('.xz') }} {{ images_dir }}/{{ item.key }}.qcow2 + loop: "{{ images | dict2items }}" + when: + - not (item.value.converted | bool) + +- name: Make sure all images are ending with qcow2 + command: cp {{ images_dir }}/{{ item.value.filename.rstrip('.xz') }} {{ images_dir }}/{{ item.key }}.qcow2 + loop: "{{ images | dict2items }}" + when: + - item.value.converted | bool + +- name: Resize images + command: qemu-img resize {{ images_dir }}/{{ item.key }}.qcow2 +8G + loop: "{{ images | dict2items }}" + +# STEP 2: Include the images inside a container +- name: Template default Dockerfile + template: + src: Dockerfile + dest: "{{ images_dir }}/Dockerfile" + mode: 0644 + +- name: Create docker images for each OS + command: docker build -t {{ registry }}/vm-{{ item.key }}:{{ item.value.tag }} --build-arg cloud_image="{{ item.key }}.qcow2" {{ images_dir }} + loop: "{{ images | dict2items }}" + +- name: Docker login + command: docker login -u="{{ docker_user }}" -p="{{ docker_password }}" "{{ docker_host }}" + +- name: Docker push image + command: docker push {{ registry }}/vm-{{ item.key }}:{{ item.value.tag }} + loop: "{{ images | dict2items }}" + +- name: Docker logout + command: docker logout -u="{{ docker_user }}" "{{ docker_host }}" diff --git a/kubespray/test-infra/image-builder/roles/kubevirt-images/templates/Dockerfile b/kubespray/test-infra/image-builder/roles/kubevirt-images/templates/Dockerfile new file mode 100644 index 0000000..f776cbf --- /dev/null +++ b/kubespray/test-infra/image-builder/roles/kubevirt-images/templates/Dockerfile @@ -0,0 +1,6 @@ +FROM kubevirt/registry-disk-v1alpha + +ARG cloud_image +MAINTAINER "The Kubespray Project" + +COPY $cloud_image /disk diff --git a/kubespray/test-infra/vagrant-docker/Dockerfile b/kubespray/test-infra/vagrant-docker/Dockerfile new file mode 100644 index 0000000..f12a740 --- /dev/null +++ b/kubespray/test-infra/vagrant-docker/Dockerfile @@ -0,0 +1,16 @@ +# Docker image published at quay.io/kubespray/vagrant + +ARG KUBESPRAY_VERSION +FROM quay.io/kubespray/kubespray:${KUBESPRAY_VERSION} + +ENV VAGRANT_VERSION=2.3.4 +ENV VAGRANT_DEFAULT_PROVIDER=libvirt +ENV VAGRANT_ANSIBLE_TAGS=facts + +RUN apt-get update && apt-get install -y wget libvirt-dev openssh-client rsync git build-essential + +# Install Vagrant +RUN wget https://releases.hashicorp.com/vagrant/${VAGRANT_VERSION}/vagrant_${VAGRANT_VERSION}-1_amd64.deb && \ + dpkg -i vagrant_${VAGRANT_VERSION}-1_amd64.deb && \ + rm vagrant_${VAGRANT_VERSION}-1_amd64.deb && \ + vagrant plugin install vagrant-libvirt diff --git a/kubespray/test-infra/vagrant-docker/README.md b/kubespray/test-infra/vagrant-docker/README.md new file mode 100644 index 0000000..36dcb9e --- /dev/null +++ b/kubespray/test-infra/vagrant-docker/README.md @@ -0,0 +1,24 @@ +# vagrant docker image + +This image is used for the vagrant CI jobs. It is using the libvirt driver. + +## Usage + +```console +$ docker run --net host --rm -it -v /var/run/libvirt/libvirt-sock:/var/run/libvirt/libvirt-sock quay.io/kubespray/vagrant +$ vagrant up +Bringing machine 'k8s-1' up with 'libvirt' provider... +Bringing machine 'k8s-2' up with 'libvirt' provider... +Bringing machine 'k8s-3' up with 'libvirt' provider... +[...] +``` + +## Cache + +You can set `/root/kubespray_cache` as a volume to keep cache between runs. + +## Building + +```shell +./build.sh v2.12.5 +``` diff --git a/kubespray/test-infra/vagrant-docker/build.sh b/kubespray/test-infra/vagrant-docker/build.sh new file mode 100755 index 0000000..dcf5445 --- /dev/null +++ b/kubespray/test-infra/vagrant-docker/build.sh @@ -0,0 +1,13 @@ +#!/bin/sh +set -euo pipefail + +if [ "$#" -ne 1 ]; then + echo "Usage: $0 tag" >&2 + exit 1 +fi + +VERSION="$1" +IMG="quay.io/kubespray/vagrant:${VERSION}" + +docker build . --build-arg "KUBESPRAY_VERSION=${VERSION}" --tag "$IMG" +docker push "$IMG" diff --git a/kubespray/tests/Makefile b/kubespray/tests/Makefile new file mode 100644 index 0000000..c9f561e --- /dev/null +++ b/kubespray/tests/Makefile @@ -0,0 +1,90 @@ +INVENTORY=$(PWD)/../inventory/sample/${CI_JOB_NAME}-${BUILD_NUMBER}.ini + +$(HOME)/.ssh/id_rsa: + mkdir -p $(HOME)/.ssh + echo $(PRIVATE_KEY) | base64 -d > $(HOME)/.ssh/id_rsa + chmod 400 $(HOME)/.ssh/id_rsa + +init-gce: $(HOME)/.ssh/id_rsa + # echo $(GCE_PEM_FILE) | base64 -d > $(HOME)/.ssh/gce + echo "$(GCE_CREDENTIALS_B64)" | base64 -d > $(HOME)/.ssh/gce.json + +init-do: $(HOME)/.ssh/id_rsa + echo $(DO_PRIVATE_KEY) | base64 -d > $(HOME)/.ssh/id_rsa + +init-packet: + echo $(PACKET_VM_SSH_PRIVATE_KEY) | base64 -d > $(HOME)/.ssh/id_rsa + chmod 400 $(HOME)/.ssh/id_rsa + +create-tf: + ./scripts/create-tf.sh + +delete-tf: + ./scripts/delete-tf.sh + +create-gce: init-gce + ansible-playbook cloud_playbooks/create-gce.yml -i local_inventory/hosts.cfg -c local \ + $(ANSIBLE_LOG_LEVEL) \ + -e @"files/${CI_JOB_NAME}.yml" \ + -e gce_credentials_file=$(HOME)/.ssh/gce.json \ + -e gce_project_id=$(GCE_PROJECT_ID) \ + -e gce_service_account_email=$(GCE_ACCOUNT) \ + -e inventory_path=$(INVENTORY) \ + -e test_id=$(TEST_ID) \ + -e preemptible=$(GCE_PREEMPTIBLE) + + +delete-gce: + ansible-playbook -i $(INVENTORY) cloud_playbooks/delete-gce.yml -c local \ + $(ANSIBLE_LOG_LEVEL) \ + -e @"files/${CI_JOB_NAME}.yml" \ + -e test_id=$(TEST_ID) \ + -e gce_project_id=$(GCE_PROJECT_ID) \ + -e gce_service_account_email=$(GCE_ACCOUNT) \ + -e gce_credentials_file=$(HOME)/.ssh/gce.json \ + -e inventory_path=$(INVENTORY) + +create-do: init-do + ansible-playbook cloud_playbooks/create-do.yml -i local_inventory/hosts.cfg -c local \ + ${ANSIBLE_LOG_LEVEL} \ + -e @"files/${CI_JOB_NAME}.yml" \ + -e inventory_path=$(INVENTORY) \ + -e test_id=${TEST_ID} + +delete-do: + ansible-playbook -i $(INVENTORY) cloud_playbooks/create-do.yml -c local \ + $(ANSIBLE_LOG_LEVEL) \ + -e @"files/${CI_JOB_NAME}.yml" \ + -e state=absent \ + -e test_id=${TEST_ID} \ + -e inventory_path=$(INVENTORY) + +create-packet: init-packet + ansible-playbook cloud_playbooks/create-packet.yml -c local \ + $(ANSIBLE_LOG_LEVEL) \ + -e @"files/${CI_JOB_NAME}.yml" \ + -e test_id=$(TEST_ID) \ + -e branch="$(CI_COMMIT_BRANCH)" \ + -e pipeline_id="$(CI_PIPELINE_ID)" \ + -e inventory_path=$(INVENTORY) + +delete-packet: + ansible-playbook cloud_playbooks/delete-packet.yml -c local \ + $(ANSIBLE_LOG_LEVEL) \ + -e @"files/${CI_JOB_NAME}.yml" \ + -e test_id=$(TEST_ID) \ + -e branch="$(CI_COMMIT_BRANCH)" \ + -e pipeline_id="$(CI_PIPELINE_ID)" \ + -e inventory_path=$(INVENTORY) + +cleanup-packet: + ansible-playbook cloud_playbooks/cleanup-packet.yml -c local \ + $(ANSIBLE_LOG_LEVEL) + +create-vagrant: + vagrant up + find / -name vagrant_ansible_inventory + cp /builds/kargo-ci/kubernetes-sigs-kubespray/inventory/sample/vagrant_ansible_inventory $(INVENTORY) + +delete-vagrant: + vagrant destroy -f diff --git a/kubespray/tests/README.md b/kubespray/tests/README.md new file mode 100644 index 0000000..05daed2 --- /dev/null +++ b/kubespray/tests/README.md @@ -0,0 +1,40 @@ +# Kubespray cloud deployment tests + +## Amazon Web Service + +| | Calico | Flannel | Weave | +------------- | ------------- | ------------- | ------------- | +Debian Jessie | [![Build Status](https://ci.kubespray.io/job/kubespray-aws-calico-jessie/badge/icon)](https://ci.kubespray.io/job/kubespray-aws-calico-jessie) | [![Build Status](https://ci.kubespray.io/job/kubespray-aws-flannel-jessie/badge/icon)](https://ci.kubespray.io/job/kubespray-aws-flannel-jessie/) | [![Build Status](https://ci.kubespray.io/job/kubespray-aws-weave-jessie/badge/icon)](https://ci.kubespray.io/job/kubespray-aws-weave-jessie/) | +Ubuntu Trusty |[![Build Status](https://ci.kubespray.io/job/kubespray-aws-calico-trusty/badge/icon)](https://ci.kubespray.io/job/kubespray-aws-calico-trusty/)|[![Build Status](https://ci.kubespray.io/job/kubespray-aws-flannel-trusty/badge/icon)](https://ci.kubespray.io/job/kubespray-aws-flannel-trusty/)|[![Build Status](https://ci.kubespray.io/job/kubespray-aws-weave-trusty/badge/icon)](https://ci.kubespray.io/job/kubespray-aws-weave-trusty)| +RHEL 7.2 |[![Build Status](https://ci.kubespray.io/job/kubespray-aws-calico-rhel72/badge/icon)](https://ci.kubespray.io/job/kubespray-aws-calico-rhel72/)|[![Build Status](https://ci.kubespray.io/job/kubespray-aws-flannel-rhel72/badge/icon)](https://ci.kubespray.io/job/kubespray-aws-flannel-rhel72/)|[![Build Status](https://ci.kubespray.io/job/kubespray-aws-weave-rhel72/badge/icon)](https://ci.kubespray.io/job/kubespray-aws-weave-rhel72/)| +CentOS 7 |[![Build Status](https://ci.kubespray.io/job/kubespray-aws-calico-centos7/badge/icon)](https://ci.kubespray.io/job/kubespray-aws-calico-centos7/)|[![Build Status](https://ci.kubespray.io/job/kubespray-aws-flannel-centos7/badge/icon)](https://ci.kubespray.io/job/kubespray-aws-flannel-centos7/)|[![Build Status](https://ci.kubespray.io/job/kubespray-aws-weave-centos7/badge/icon)](https://ci.kubespray.io/job/kubespray-aws-weave-centos7/)| + +## Test environment variables + +### Common + +Variable | Description | Required | Default +--------------------- | -------------------------------------- | ---------- | -------- +`TEST_ID` | A unique execution ID for this test | Yes | +`KUBE_NETWORK_PLUGIN` | The network plugin (calico or flannel) | Yes | +`PRIVATE_KEY_FILE` | The path to the SSH private key file | No | + +### AWS Tests + +Variable | Description | Required | Default +--------------------- | ----------------------------------------------- | ---------- | --------- +`AWS_ACCESS_KEY` | The Amazon Access Key ID | Yes | +`AWS_SECRET_KEY` | The Amazon Secret Access Key | Yes | +`AWS_AMI_ID` | The AMI ID to deploy | Yes | +`AWS_KEY_PAIR_NAME` | The name of the EC2 key pair to use | Yes | +`AWS_SECURITY_GROUP` | The EC2 Security Group to use | No | default +`AWS_REGION` | The EC2 region | No | eu-central-1 + +#### Use private ssh key + +##### Key + +```bash +openssl pkcs12 -in gce-secure.p12 -passin pass:notasecret -nodes -nocerts | openssl rsa -out gce-secure.pem +cat gce-secure.pem |base64 -w0 > GCE_PEM_FILE` +``` diff --git a/kubespray/tests/ansible.cfg b/kubespray/tests/ansible.cfg new file mode 100644 index 0000000..88531be --- /dev/null +++ b/kubespray/tests/ansible.cfg @@ -0,0 +1,15 @@ +[ssh_connection] +pipelining=True +ansible_ssh_common_args = -o ControlMaster=auto -o ControlPersist=30m -o ConnectionAttempts=100 +retries=2 +[defaults] +forks = 20 +host_key_checking=False +gathering = smart +fact_caching = jsonfile +fact_caching_connection = /tmp +stdout_callback = skippy +library = ./library:../library +callbacks_enabled = profile_tasks +jinja2_extensions = jinja2.ext.do +roles_path = ../roles diff --git a/kubespray/tests/cloud_playbooks/cleanup-packet.yml b/kubespray/tests/cloud_playbooks/cleanup-packet.yml new file mode 100644 index 0000000..009071e --- /dev/null +++ b/kubespray/tests/cloud_playbooks/cleanup-packet.yml @@ -0,0 +1,8 @@ +--- + +- name: Cleanup packet vms + hosts: localhost + gather_facts: no + become: true + roles: + - { role: cleanup-packet-ci } diff --git a/kubespray/tests/cloud_playbooks/create-aws.yml b/kubespray/tests/cloud_playbooks/create-aws.yml new file mode 100644 index 0000000..3a31d29 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/create-aws.yml @@ -0,0 +1,26 @@ +--- +- name: Provision AWS VMs + hosts: localhost + become: False + gather_facts: False + + tasks: + - name: Provision a set of instances + amazon.aws.ec2_instance: + key_name: "{{ aws.key_name }}" + aws_access_key: "{{ aws.access_key }}" + aws_secret_key: "{{ aws.secret_key }}" + region: "{{ aws.region }}" + group_id: "{{ aws.group }}" + instance_type: "{{ aws.instance_type }}" + image: "{{ aws.ami_id }}" + wait: true + count: "{{ aws.count }}" + instance_tags: "{{ aws.tags }}" + register: ec2 + + - name: Template the inventory + template: + src: ../templates/inventory-aws.j2 # noqa no-relative-paths - CI inventory templates are not in role_path + dest: "{{ inventory_path }}" + mode: 0644 diff --git a/kubespray/tests/cloud_playbooks/create-do.yml b/kubespray/tests/cloud_playbooks/create-do.yml new file mode 100644 index 0000000..3c25062 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/create-do.yml @@ -0,0 +1,94 @@ +--- +- name: Provision Digital Ocean VMs + hosts: localhost + become: false + gather_facts: no + vars: + state: "present" + ssh_key_id: "6536865" + cloud_machine_type: 2gb + regions: + - nyc1 + - sfo1 + - nyc2 + - ams2 + - sgp1 + - lon1 + - nyc3 + - ams3 + - fra1 + - tor1 + - sfo2 + - blr1 + cloud_images: + - fedora-24-x64 + - centos-5-x64 + - centos-5-x32 + - fedora-25-x64 + - debian-7-x64 + - debian-7-x32 + - debian-8-x64 + - debian-8-x32 + - centos-6-x32 + - centos-6-x64 + - ubuntu-16-10-x32 + - ubuntu-16-10-x64 + - freebsd-11-0-x64-zfs + - freebsd-10-3-x64-zfs + - ubuntu-12-04-x32 + - ubuntu-12-04-x64 + - ubuntu-16-04-x64 + - ubuntu-16-04-x32 + - ubuntu-14-04-x64 + - ubuntu-14-04-x32 + - centos-7-x64 + - freebsd-11-0-x64 + - freebsd-10-3-x64 + - centos-7-3-1611-x64 + mode: default + + tasks: + - name: Replace_test_id + set_fact: + test_name: "{{ test_id | regex_replace('\\.', '-') }}" + + - name: Show vars + debug: + msg: "{{ cloud_region }}, {{ cloud_image }}" + + - name: Set instance names + set_fact: + # noqa: jinja[spacing] + instance_names: >- + {%- if mode in ['separate', 'ha'] -%} + ["k8s-{{ test_name }}-1", "k8s-{{ test_name }}-2", "k8s-{{ test_name }}-3"] + {%- else -%} + ["k8s-{{ test_name }}-1", "k8s-{{ test_name }}-2"] + {%- endif -%} + + - name: Manage DO instances | {{ state }} + community.digitalocean.digital_ocean: + unique_name: yes + api_token: "{{ lookup('env', 'DO_API_TOKEN') }}" + command: "droplet" + image_id: "{{ cloud_image }}" + name: "{{ item }}" + private_networking: no + region_id: "{{ cloud_region }}" + size_id: "{{ cloud_machine_type }}" + ssh_key_ids: "{{ ssh_key_id }}" + state: "{{ state }}" + wait: yes + register: droplets + with_items: "{{ instance_names }}" + + - debug: # noqa unnamed-task + msg: "{{ droplets }}, {{ inventory_path }}" + when: state == 'present' + + - name: Template the inventory + template: + src: ../templates/inventory-do.j2 # noqa no-relative-paths - CI templates are not in role_path + dest: "{{ inventory_path }}" + mode: 0644 + when: state == 'present' diff --git a/kubespray/tests/cloud_playbooks/create-gce.yml b/kubespray/tests/cloud_playbooks/create-gce.yml new file mode 100644 index 0000000..78c96b0 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/create-gce.yml @@ -0,0 +1,81 @@ +--- +- name: Provision Google Cloud VMs + hosts: localhost + become: false + gather_facts: no + vars: + cloud_machine_type: g1-small + mode: default + preemptible: no + ci_job_name: "{{ lookup('env', 'CI_JOB_NAME') }}" + delete_group_vars: no + tasks: + - name: Include vars for test {{ ci_job_name }} + include_vars: "../files/{{ ci_job_name }}.yml" + + - name: Replace_test_id + set_fact: + test_name: "{{ test_id | regex_replace('\\.', '-') }}" + + - name: Set instance names + set_fact: + # noqa: jinja[spacing] + instance_names: >- + {%- if mode in ['separate', 'separate-scale', 'ha', 'ha-scale'] -%} + k8s-{{ test_name }}-1,k8s-{{ test_name }}-2,k8s-{{ test_name }}-3 + {%- elif mode == 'aio' -%} + k8s-{{ test_name }}-1 + {%- else -%} + k8s-{{ test_name }}-1,k8s-{{ test_name }}-2 + {%- endif -%} + + - name: Create gce instances + google.cloud.gcp_compute_instance: # noqa args[module] - Probably doesn't work + instance_names: "{{ instance_names }}" + machine_type: "{{ cloud_machine_type }}" + image: "{{ cloud_image | default(omit) }}" + image_family: "{{ cloud_image_family | default(omit) }}" + preemptible: "{{ preemptible }}" + service_account_email: "{{ gce_service_account_email }}" + pem_file: "{{ gce_pem_file | default(omit) }}" + credentials_file: "{{ gce_credentials_file | default(omit) }}" + project_id: "{{ gce_project_id }}" + zone: "{{ cloud_region }}" + metadata: '{"test_id": "{{ test_id }}", "network": "{{ kube_network_plugin }}", "startup-script": "{{ startup_script | default("") }}"}' + tags: "build-{{ test_name }},{{ kube_network_plugin }}" + ip_forward: yes + service_account_permissions: ['compute-rw'] + register: gce + + - name: Add instances to host group + add_host: + hostname: "{{ item.public_ip }}" + groupname: "waitfor_hosts" + with_items: '{{ gce.instance_data }}' + + - name: Template the inventory # noqa no-relative-paths - CI inventory templates are not in role_path + template: + src: ../templates/inventory-gce.j2 + dest: "{{ inventory_path }}" + mode: 0644 + + - name: Make group_vars directory + file: + path: "{{ inventory_path | dirname }}/group_vars" + state: directory + mode: 0755 + when: mode in ['scale', 'separate-scale', 'ha-scale'] + + - name: Template fake hosts group vars # noqa no-relative-paths - CI templates are not in role_path + template: + src: ../templates/fake_hosts.yml.j2 + dest: "{{ inventory_path | dirname }}/group_vars/fake_hosts.yml" + mode: 0644 + when: mode in ['scale', 'separate-scale', 'ha-scale'] + + - name: Delete group_vars directory + file: + path: "{{ inventory_path | dirname }}/group_vars" + state: absent + recurse: yes + when: delete_group_vars diff --git a/kubespray/tests/cloud_playbooks/create-packet.yml b/kubespray/tests/cloud_playbooks/create-packet.yml new file mode 100644 index 0000000..8212fb6 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/create-packet.yml @@ -0,0 +1,11 @@ +--- + +- name: Provision Packet VMs + hosts: localhost + gather_facts: no + become: true + vars: + ci_job_name: "{{ lookup('env', 'CI_JOB_NAME') }}" + test_name: "{{ test_id | regex_replace('\\.', '-') }}" + roles: + - { role: packet-ci, vm_cleanup: false } diff --git a/kubespray/tests/cloud_playbooks/delete-aws.yml b/kubespray/tests/cloud_playbooks/delete-aws.yml new file mode 100644 index 0000000..cd5a200 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/delete-aws.yml @@ -0,0 +1,19 @@ +--- +- name: Terminate AWS VMs + hosts: kube_node + become: False + + tasks: + - name: Gather EC2 facts + amazon.aws.ec2_metadata_facts: + + - name: Terminate EC2 instances + amazon.aws.ec2_instance: + aws_access_key: "{{ aws_access_key }}" + aws_secret_key: "{{ aws_secret_key }}" + state: absent + instance_ids: "{{ ansible_ec2_instance_id }}" + region: "{{ ansible_ec2_placement_region }}" + wait: True + delegate_to: localhost + connection: local diff --git a/kubespray/tests/cloud_playbooks/delete-gce.yml b/kubespray/tests/cloud_playbooks/delete-gce.yml new file mode 100644 index 0000000..8752f24 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/delete-gce.yml @@ -0,0 +1,50 @@ +--- +- name: Terminate Google Cloud VMs + hosts: localhost + become: false + gather_facts: no + vars: + mode: default + + tasks: + - name: Replace_test_id + set_fact: + test_name: "{{ test_id | regex_replace('\\.', '-') }}" + + - name: Set instance names + set_fact: + # noqa: jinja[spacing] + instance_names: >- + {%- if mode in ['separate', 'ha'] -%} + k8s-{{ test_name }}-1,k8s-{{ test_name }}-2,k8s-{{ test_name }}-3 + {%- else -%} + k8s-{{ test_name }}-1,k8s-{{ test_name }}-2 + {%- endif -%} + + - name: Stop gce instances # noqa args[module] - Probably doesn't work + google.cloud.gcp_compute_instance: + instance_names: "{{ instance_names }}" + image: "{{ cloud_image | default(omit) }}" + service_account_email: "{{ gce_service_account_email }}" + pem_file: "{{ gce_pem_file | default(omit) }}" + credentials_file: "{{ gce_credentials_file | default(omit) }}" + project_id: "{{ gce_project_id }}" + zone: "{{ cloud_region | default('europe-west1-b') }}" + state: 'stopped' + async: 120 + poll: 3 + register: gce + + - name: Delete gce instances # noqa args[module] - Probably doesn't work + google.cloud.gcp_compute_instance: + instance_names: "{{ instance_names }}" + image: "{{ cloud_image | default(omit) }}" + service_account_email: "{{ gce_service_account_email }}" + pem_file: "{{ gce_pem_file | default(omit) }}" + credentials_file: "{{ gce_credentials_file | default(omit) }}" + project_id: "{{ gce_project_id }}" + zone: "{{ cloud_region | default('europe-west1-b') }}" + state: 'absent' + async: 120 + poll: 3 + register: gce diff --git a/kubespray/tests/cloud_playbooks/delete-packet.yml b/kubespray/tests/cloud_playbooks/delete-packet.yml new file mode 100644 index 0000000..7d0c900 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/delete-packet.yml @@ -0,0 +1,11 @@ +--- + +- name: Terminate Packet VMs + hosts: localhost + gather_facts: no + become: true + vars: + ci_job_name: "{{ lookup('env', 'CI_JOB_NAME') }}" + test_name: "{{ test_id | regex_replace('\\.', '-') }}" + roles: + - { role: packet-ci, vm_cleanup: true } diff --git a/kubespray/tests/cloud_playbooks/roles/cleanup-packet-ci/tasks/main.yml b/kubespray/tests/cloud_playbooks/roles/cleanup-packet-ci/tasks/main.yml new file mode 100644 index 0000000..9256b2d --- /dev/null +++ b/kubespray/tests/cloud_playbooks/roles/cleanup-packet-ci/tasks/main.yml @@ -0,0 +1,16 @@ +--- + +- name: Fetch a list of namespaces + kubernetes.core.k8s_info: + api_version: v1 + kind: Namespace + label_selectors: + - cijobs = true + register: namespaces + +- name: Delete stale namespaces for more than 2 hours + command: "kubectl delete namespace {{ item.metadata.name }}" + failed_when: false + loop: "{{ namespaces.resources }}" + when: + - (now() - (item.metadata.creationTimestamp | to_datetime("%Y-%m-%dT%H:%M:%SZ"))).total_seconds() >= 7200 diff --git a/kubespray/tests/cloud_playbooks/roles/packet-ci/defaults/main.yml b/kubespray/tests/cloud_playbooks/roles/packet-ci/defaults/main.yml new file mode 100644 index 0000000..17dd3d8 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/roles/packet-ci/defaults/main.yml @@ -0,0 +1,44 @@ +--- + +# VM sizing +vm_cpu_cores: 2 +vm_cpu_sockets: 1 +vm_cpu_threads: 2 +vm_memory: 2048Mi + +# Replace invalid characters so that we can use the branch name in kubernetes labels +branch_name_sane: "{{ branch | regex_replace('/', '-') }}" + +# Request/Limit allocation settings + +cpu_allocation_ratio: 0.5 +memory_allocation_ratio: 1 + +# Default path for inventory +inventory_path: "/tmp/{{ test_name }}/inventory" + +# Deployment mode +mode: aio + +# Cloud init config for each os type +# distro: fedora -> I2Nsb3VkLWNvbmZpZwpzeXN0ZW1faW5mbzoKICBkaXN0cm86IGZlZG9yYQp1c2VyczoKIC0gbmFtZToga3ViZXNwcmF5CiAgIGdyb3Vwczogd2hlZWwKICAgc3VkbzogJ0FMTD0oQUxMKSBOT1BBU1NXRDpBTEwnCiAgIHNoZWxsOiAvYmluL2Jhc2gKICAgbG9ja19wYXNzd2Q6IEZhbHNlCiAgIGhvbWU6IC9ob21lL2t1YmVzcHJheQogICBzc2hfYXV0aG9yaXplZF9rZXlzOgogICAgIC0gc3NoLXJzYSBBQUFBQjNOemFDMXljMkVBQUFBREFRQUJBQUFCQVFDYW5UaS9lS3gwK3RIWUpBZURocStzRlMyT2JVUDEvSTY5ZjdpVjNVdGtLbFQyMEpmVzFmNkZlWHQvMDRWZjI3V1FxK05xczZ2R0JxRDlRWFNZdWYrdDAvczdFUExqVGVpOW1lMW1wcXIrdVRlK0tEdFRQMzlwZkQzL2VWQ2FlQjcyNkdQMkZrYUQwRnpwbUViNjZPM05xaHhPUTk2R3gvOVhUdXcvSzNsbGo0T1ZENkdyalIzQjdjNFh0RUJzWmNacHBNSi9vSDFtR3lHWGRoMzFtV1FTcUFSTy9QOFU4R3d0MCtIR3BVd2gvaGR5M3QrU1lvVEIyR3dWYjB6b3lWd3RWdmZEUXpzbThmcTNhdjRLdmV6OGtZdU5ESnYwNXg0bHZVWmdSMTVaRFJYc0FuZGhReXFvWGRDTEFlMCtlYUtYcTlCa1d4S0ZiOWhQZTBBVWpqYTU= +# distro: rhel: -> I2Nsb3VkLWNvbmZpZwpzeXN0ZW1faW5mbzoKICBkaXN0cm86IHJoZWwKdXNlcnM6CiAtIG5hbWU6IGt1YmVzcHJheQogICBncm91cHM6IHdoZWVsCiAgIHN1ZG86ICdBTEw9KEFMTCkgTk9QQVNTV0Q6QUxMJwogICBzaGVsbDogL2Jpbi9iYXNoCiAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICBob21lOiAvaG9tZS9rdWJlc3ByYXkKICAgc3NoX2F1dGhvcml6ZWRfa2V5czoKICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1Cgo= +# distro: rhel (+ sudo and hostname packages): -> I2Nsb3VkLWNvbmZpZwpwYWNrYWdlczoKIC0gc3VkbwogLSBob3N0bmFtZQpzeXN0ZW1faW5mbzoKICBkaXN0cm86IHJoZWwKdXNlcnM6CiAtIG5hbWU6IGt1YmVzcHJheQogICBncm91cHM6IHdoZWVsCiAgIHN1ZG86ICdBTEw9KEFMTCkgTk9QQVNTV0Q6QUxMJwogICBzaGVsbDogL2Jpbi9iYXNoCiAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICBob21lOiAvaG9tZS9rdWJlc3ByYXkKICAgc3NoX2F1dGhvcml6ZWRfa2V5czoKICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1Cgo= +# generic one -> I2Nsb3VkLWNvbmZpZwogdXNlcnM6CiAgLSBuYW1lOiBrdWJlc3ByYXkKICAgIHN1ZG86IEFMTD0oQUxMKSBOT1BBU1NXRDpBTEwKICAgIHNoZWxsOiAvYmluL2Jhc2gKICAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICAgaG9tZTogL2hvbWUva3ViZXNwcmF5CiAgICBzc2hfYXV0aG9yaXplZF9rZXlzOgogICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1 +cloud_init: + centos-7: "I2Nsb3VkLWNvbmZpZwpzeXN0ZW1faW5mbzoKICBkaXN0cm86IHJoZWwKdXNlcnM6CiAtIG5hbWU6IGt1YmVzcHJheQogICBncm91cHM6IHdoZWVsCiAgIHN1ZG86ICdBTEw9KEFMTCkgTk9QQVNTV0Q6QUxMJwogICBzaGVsbDogL2Jpbi9iYXNoCiAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICBob21lOiAvaG9tZS9rdWJlc3ByYXkKICAgc3NoX2F1dGhvcml6ZWRfa2V5czoKICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1Cgo=" + centos-8: "I2Nsb3VkLWNvbmZpZwpzeXN0ZW1faW5mbzoKICBkaXN0cm86IHJoZWwKdXNlcnM6CiAtIG5hbWU6IGt1YmVzcHJheQogICBncm91cHM6IHdoZWVsCiAgIHN1ZG86ICdBTEw9KEFMTCkgTk9QQVNTV0Q6QUxMJwogICBzaGVsbDogL2Jpbi9iYXNoCiAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICBob21lOiAvaG9tZS9rdWJlc3ByYXkKICAgc3NoX2F1dGhvcml6ZWRfa2V5czoKICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1Cgo=" + almalinux-8: "I2Nsb3VkLWNvbmZpZwpzeXN0ZW1faW5mbzoKICBkaXN0cm86IHJoZWwKdXNlcnM6CiAtIG5hbWU6IGt1YmVzcHJheQogICBncm91cHM6IHdoZWVsCiAgIHN1ZG86ICdBTEw9KEFMTCkgTk9QQVNTV0Q6QUxMJwogICBzaGVsbDogL2Jpbi9iYXNoCiAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICBob21lOiAvaG9tZS9rdWJlc3ByYXkKICAgc3NoX2F1dGhvcml6ZWRfa2V5czoKICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1Cgo=" + rockylinux-8: "I2Nsb3VkLWNvbmZpZwpwYWNrYWdlczoKIC0gc3VkbwogLSBob3N0bmFtZQpzeXN0ZW1faW5mbzoKICBkaXN0cm86IHJoZWwKdXNlcnM6CiAtIG5hbWU6IGt1YmVzcHJheQogICBncm91cHM6IHdoZWVsCiAgIHN1ZG86ICdBTEw9KEFMTCkgTk9QQVNTV0Q6QUxMJwogICBzaGVsbDogL2Jpbi9iYXNoCiAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICBob21lOiAvaG9tZS9rdWJlc3ByYXkKICAgc3NoX2F1dGhvcml6ZWRfa2V5czoKICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1Cgo=" + rockylinux-9: "I2Nsb3VkLWNvbmZpZwpwYWNrYWdlczoKIC0gc3VkbwogLSBob3N0bmFtZQpzeXN0ZW1faW5mbzoKICBkaXN0cm86IHJoZWwKdXNlcnM6CiAtIG5hbWU6IGt1YmVzcHJheQogICBncm91cHM6IHdoZWVsCiAgIHN1ZG86ICdBTEw9KEFMTCkgTk9QQVNTV0Q6QUxMJwogICBzaGVsbDogL2Jpbi9iYXNoCiAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICBob21lOiAvaG9tZS9rdWJlc3ByYXkKICAgc3NoX2F1dGhvcml6ZWRfa2V5czoKICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1Cgo=" + debian-10: "I2Nsb3VkLWNvbmZpZwogdXNlcnM6CiAgLSBuYW1lOiBrdWJlc3ByYXkKICAgIHN1ZG86IEFMTD0oQUxMKSBOT1BBU1NXRDpBTEwKICAgIHNoZWxsOiAvYmluL2Jhc2gKICAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICAgaG9tZTogL2hvbWUva3ViZXNwcmF5CiAgICBzc2hfYXV0aG9yaXplZF9rZXlzOgogICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1" + debian-11: "I2Nsb3VkLWNvbmZpZwogdXNlcnM6CiAgLSBuYW1lOiBrdWJlc3ByYXkKICAgIHN1ZG86IEFMTD0oQUxMKSBOT1BBU1NXRDpBTEwKICAgIHNoZWxsOiAvYmluL2Jhc2gKICAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICAgaG9tZTogL2hvbWUva3ViZXNwcmF5CiAgICBzc2hfYXV0aG9yaXplZF9rZXlzOgogICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1" + debian-12: "I2Nsb3VkLWNvbmZpZwogdXNlcnM6CiAgLSBuYW1lOiBrdWJlc3ByYXkKICAgIHN1ZG86IEFMTD0oQUxMKSBOT1BBU1NXRDpBTEwKICAgIHNoZWxsOiAvYmluL2Jhc2gKICAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICAgaG9tZTogL2hvbWUva3ViZXNwcmF5CiAgICBzc2hfYXV0aG9yaXplZF9rZXlzOgogICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1" + fedora-37: "I2Nsb3VkLWNvbmZpZwpzeXN0ZW1faW5mbzoKICBkaXN0cm86IGZlZG9yYQp1c2VyczoKIC0gbmFtZToga3ViZXNwcmF5CiAgIGdyb3Vwczogd2hlZWwKICAgc3VkbzogJ0FMTD0oQUxMKSBOT1BBU1NXRDpBTEwnCiAgIHNoZWxsOiAvYmluL2Jhc2gKICAgbG9ja19wYXNzd2Q6IEZhbHNlCiAgIGhvbWU6IC9ob21lL2t1YmVzcHJheQogICBzc2hfYXV0aG9yaXplZF9rZXlzOgogICAgIC0gc3NoLXJzYSBBQUFBQjNOemFDMXljMkVBQUFBREFRQUJBQUFCQVFDYW5UaS9lS3gwK3RIWUpBZURocStzRlMyT2JVUDEvSTY5ZjdpVjNVdGtLbFQyMEpmVzFmNkZlWHQvMDRWZjI3V1FxK05xczZ2R0JxRDlRWFNZdWYrdDAvczdFUExqVGVpOW1lMW1wcXIrdVRlK0tEdFRQMzlwZkQzL2VWQ2FlQjcyNkdQMkZrYUQwRnpwbUViNjZPM05xaHhPUTk2R3gvOVhUdXcvSzNsbGo0T1ZENkdyalIzQjdjNFh0RUJzWmNacHBNSi9vSDFtR3lHWGRoMzFtV1FTcUFSTy9QOFU4R3d0MCtIR3BVd2gvaGR5M3QrU1lvVEIyR3dWYjB6b3lWd3RWdmZEUXpzbThmcTNhdjRLdmV6OGtZdU5ESnYwNXg0bHZVWmdSMTVaRFJYc0FuZGhReXFvWGRDTEFlMCtlYUtYcTlCa1d4S0ZiOWhQZTBBVWpqYTU=" + fedora-38: "I2Nsb3VkLWNvbmZpZwpzeXN0ZW1faW5mbzoKICBkaXN0cm86IGZlZG9yYQp1c2VyczoKIC0gbmFtZToga3ViZXNwcmF5CiAgIGdyb3Vwczogd2hlZWwKICAgc3VkbzogJ0FMTD0oQUxMKSBOT1BBU1NXRDpBTEwnCiAgIHNoZWxsOiAvYmluL2Jhc2gKICAgbG9ja19wYXNzd2Q6IEZhbHNlCiAgIGhvbWU6IC9ob21lL2t1YmVzcHJheQogICBzc2hfYXV0aG9yaXplZF9rZXlzOgogICAgIC0gc3NoLXJzYSBBQUFBQjNOemFDMXljMkVBQUFBREFRQUJBQUFCQVFDYW5UaS9lS3gwK3RIWUpBZURocStzRlMyT2JVUDEvSTY5ZjdpVjNVdGtLbFQyMEpmVzFmNkZlWHQvMDRWZjI3V1FxK05xczZ2R0JxRDlRWFNZdWYrdDAvczdFUExqVGVpOW1lMW1wcXIrdVRlK0tEdFRQMzlwZkQzL2VWQ2FlQjcyNkdQMkZrYUQwRnpwbUViNjZPM05xaHhPUTk2R3gvOVhUdXcvSzNsbGo0T1ZENkdyalIzQjdjNFh0RUJzWmNacHBNSi9vSDFtR3lHWGRoMzFtV1FTcUFSTy9QOFU4R3d0MCtIR3BVd2gvaGR5M3QrU1lvVEIyR3dWYjB6b3lWd3RWdmZEUXpzbThmcTNhdjRLdmV6OGtZdU5ESnYwNXg0bHZVWmdSMTVaRFJYc0FuZGhReXFvWGRDTEFlMCtlYUtYcTlCa1d4S0ZiOWhQZTBBVWpqYTU=" + opensuse-leap-15: "I2Nsb3VkLWNvbmZpZwogdXNlcnM6CiAgLSBuYW1lOiBrdWJlc3ByYXkKICAgIHN1ZG86IEFMTD0oQUxMKSBOT1BBU1NXRDpBTEwKICAgIHNoZWxsOiAvYmluL2Jhc2gKICAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICAgaG9tZTogL2hvbWUva3ViZXNwcmF5CiAgICBzc2hfYXV0aG9yaXplZF9rZXlzOgogICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1" + rhel-server-7: "I2Nsb3VkLWNvbmZpZwpzeXN0ZW1faW5mbzoKICBkaXN0cm86IHJoZWwKdXNlcnM6CiAtIG5hbWU6IGt1YmVzcHJheQogICBncm91cHM6IHdoZWVsCiAgIHN1ZG86ICdBTEw9KEFMTCkgTk9QQVNTV0Q6QUxMJwogICBzaGVsbDogL2Jpbi9iYXNoCiAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICBob21lOiAvaG9tZS9rdWJlc3ByYXkKICAgc3NoX2F1dGhvcml6ZWRfa2V5czoKICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1Cgo=" + amazon-linux-2: "I2Nsb3VkLWNvbmZpZwpzeXN0ZW1faW5mbzoKICBkaXN0cm86IHJoZWwKdXNlcnM6CiAtIG5hbWU6IGt1YmVzcHJheQogICBncm91cHM6IHdoZWVsCiAgIHN1ZG86ICdBTEw9KEFMTCkgTk9QQVNTV0Q6QUxMJwogICBzaGVsbDogL2Jpbi9iYXNoCiAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICBob21lOiAvaG9tZS9rdWJlc3ByYXkKICAgc3NoX2F1dGhvcml6ZWRfa2V5czoKICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1Cgo=" + ubuntu-2004: "I2Nsb3VkLWNvbmZpZwogdXNlcnM6CiAgLSBuYW1lOiBrdWJlc3ByYXkKICAgIHN1ZG86IEFMTD0oQUxMKSBOT1BBU1NXRDpBTEwKICAgIHNoZWxsOiAvYmluL2Jhc2gKICAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICAgaG9tZTogL2hvbWUva3ViZXNwcmF5CiAgICBzc2hfYXV0aG9yaXplZF9rZXlzOgogICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1" + ubuntu-2204: "I2Nsb3VkLWNvbmZpZwogdXNlcnM6CiAgLSBuYW1lOiBrdWJlc3ByYXkKICAgIHN1ZG86IEFMTD0oQUxMKSBOT1BBU1NXRDpBTEwKICAgIHNoZWxsOiAvYmluL2Jhc2gKICAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICAgaG9tZTogL2hvbWUva3ViZXNwcmF5CiAgICBzc2hfYXV0aG9yaXplZF9rZXlzOgogICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1" + oracle-7: "I2Nsb3VkLWNvbmZpZwpzeXN0ZW1faW5mbzoKICBkaXN0cm86IHJoZWwKdXNlcnM6CiAtIG5hbWU6IGt1YmVzcHJheQogICBncm91cHM6IHdoZWVsCiAgIHN1ZG86ICdBTEw9KEFMTCkgTk9QQVNTV0Q6QUxMJwogICBzaGVsbDogL2Jpbi9iYXNoCiAgIGxvY2tfcGFzc3dkOiBGYWxzZQogICBob21lOiAvaG9tZS9rdWJlc3ByYXkKICAgc3NoX2F1dGhvcml6ZWRfa2V5czoKICAgICAtIHNzaC1yc2EgQUFBQUIzTnphQzF5YzJFQUFBQURBUUFCQUFBQkFRQ2FuVGkvZUt4MCt0SFlKQWVEaHErc0ZTMk9iVVAxL0k2OWY3aVYzVXRrS2xUMjBKZlcxZjZGZVh0LzA0VmYyN1dRcStOcXM2dkdCcUQ5UVhTWXVmK3QwL3M3RVBMalRlaTltZTFtcHFyK3VUZStLRHRUUDM5cGZEMy9lVkNhZUI3MjZHUDJGa2FEMEZ6cG1FYjY2TzNOcWh4T1E5Nkd4LzlYVHV3L0szbGxqNE9WRDZHcmpSM0I3YzRYdEVCc1pjWnBwTUovb0gxbUd5R1hkaDMxbVdRU3FBUk8vUDhVOEd3dDArSEdwVXdoL2hkeTN0K1NZb1RCMkd3VmIwem95Vnd0VnZmRFF6c204ZnEzYXY0S3ZlejhrWXVOREp2MDV4NGx2VVpnUjE1WkRSWHNBbmRoUXlxb1hkQ0xBZTArZWFLWHE5QmtXeEtGYjloUGUwQVVqamE1Cgo=" diff --git a/kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/cleanup-old-vms.yml b/kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/cleanup-old-vms.yml new file mode 100644 index 0000000..052a44f --- /dev/null +++ b/kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/cleanup-old-vms.yml @@ -0,0 +1,17 @@ +--- + +- name: Fetch a list of namespaces + kubernetes.core.k8s_info: + api_version: v1 + kind: Namespace + label_selectors: + - cijobs = true + - branch = {{ branch_name_sane }} + register: namespaces + +- name: Delete older namespaces + command: "kubectl delete namespace {{ item.metadata.name }}" + failed_when: false + loop: "{{ namespaces.resources }}" + when: + - (item.metadata.labels.pipeline_id | int) < (pipeline_id | int) diff --git a/kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/create-vms.yml b/kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/create-vms.yml new file mode 100644 index 0000000..4070b24 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/create-vms.yml @@ -0,0 +1,50 @@ +--- + +- name: "Create CI namespace {{ test_name }} for test vms" + shell: |- + kubectl create namespace {{ test_name }} && + kubectl label namespace {{ test_name }} cijobs=true branch="{{ branch_name_sane }}" pipeline_id="{{ pipeline_id }}" + changed_when: false + +- name: "Create temp dir /tmp/{{ test_name }} for CI files" + file: + path: "/tmp/{{ test_name }}" + state: directory + mode: 0755 + +- name: Template vm files for CI job + set_fact: + vms_files: "{{ vms_files + [lookup('ansible.builtin.template', 'vm.yml.j2') | from_yaml] }}" + vars: + vms_files: [] + loop: "{{ range(1, vm_count | int + 1, 1) | list }}" + loop_control: + index_var: vm_id + +- name: Start vms for CI job + kubernetes.core.k8s: + definition: "{{ item }}" + changed_when: false + loop: "{{ vms_files }}" + +- name: Wait for vms to have ipaddress assigned + shell: "set -o pipefail && kubectl get vmis -n {{ test_name }} instance-{{ vm_id }} -o json | jq '.status.interfaces[].ipAddress' | tr -d '\"'" + args: + executable: /bin/bash + changed_when: false + register: vm_ips + loop: "{{ range(1, vm_count | int + 1, 1) | list }}" + loop_control: + index_var: vm_id + retries: 20 + delay: 15 + until: + - vm_ips.stdout | ipaddr + +- name: "Create inventory for CI test in file /tmp/{{ test_name }}/inventory" + template: + src: "inventory.j2" + dest: "{{ inventory_path }}" + mode: 0644 + vars: + vms: "{{ vm_ips }}" diff --git a/kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/delete-vms.yml b/kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/delete-vms.yml new file mode 100644 index 0000000..98bd05a --- /dev/null +++ b/kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/delete-vms.yml @@ -0,0 +1,30 @@ +--- + +- name: Check if temp directory for {{ test_name }} exists + stat: + path: "/tmp/{{ test_name }}" + get_attributes: no + get_checksum: no + get_mime: no + register: temp_dir_details + +- name: "Cleanup temp directory for {{ test_name }}" + file: + path: "/tmp/{{ test_name }}" + state: absent + +- name: "Cleanup namespace for {{ test_name }}" + command: "kubectl delete namespace {{ test_name }}" + changed_when: false + +- name: Wait for namespace {{ test_name }} to be fully deleted + command: kubectl get ns {{ test_name }} + register: delete_namespace + failed_when: + - delete_namespace.rc == 0 + changed_when: + - delete_namespace.rc == 0 + retries: 12 + delay: 10 + until: + - delete_namespace.rc != 0 diff --git a/kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/main.yml b/kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/main.yml new file mode 100644 index 0000000..633c872 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/roles/packet-ci/tasks/main.yml @@ -0,0 +1,21 @@ +--- + +- name: "Include custom vars for ci job: {{ ci_job_name }}" + include_vars: "../files/{{ ci_job_name }}.yml" + +- name: Set VM count needed for CI test_id + set_fact: + vm_count: "{%- if mode in ['separate', 'separate-scale', 'ha', 'ha-scale', 'ha-recover', 'ha-recover-noquorum'] -%}{{ 3 | int }}{%- elif mode == 'aio' -%}{{ 1 | int }}{%- else -%}{{ 2 | int }}{%- endif -%}" + +- name: Cleamup old VMs + import_tasks: cleanup-old-vms.yml + +- name: Create VMs + import_tasks: create-vms.yml + when: + - not vm_cleanup + +- name: Delete VMs + import_tasks: delete-vms.yml + when: + - vm_cleanup | default(false) diff --git a/kubespray/tests/cloud_playbooks/roles/packet-ci/templates/inventory.j2 b/kubespray/tests/cloud_playbooks/roles/packet-ci/templates/inventory.j2 new file mode 100644 index 0000000..c49d582 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/roles/packet-ci/templates/inventory.j2 @@ -0,0 +1,93 @@ +[all] +{% for instance in vms.results %} +instance-{{ loop.index }} ansible_host={{instance.stdout}} +{% endfor %} + +{% if mode is defined and mode in ["separate", "separate-scale"] %} +[kube_control_plane] +instance-1 + +[kube_node] +instance-2 + +[etcd] +instance-3 +{% elif mode is defined and mode in ["ha", "ha-scale"] %} +[kube_control_plane] +instance-1 +instance-2 + +[kube_node] +instance-3 + +[etcd] +instance-1 +instance-2 +instance-3 +{% elif mode == "default" %} +[kube_control_plane] +instance-1 + +[kube_node] +instance-2 + +[etcd] +instance-1 +{% elif mode == "aio" %} +[kube_control_plane] +instance-1 + +[kube_node] +instance-1 + +[etcd] +instance-1 +{% elif mode == "ha-recover" %} +[kube_control_plane] +instance-1 +instance-2 + +[kube_node] +instance-3 + +[etcd] +instance-3 +instance-1 +instance-2 + +[broken_kube_control_plane] +instance-2 + +[broken_etcd] +instance-2 etcd_member_name=etcd3 +{% elif mode == "ha-recover-noquorum" %} +[kube_control_plane] +instance-3 +instance-1 +instance-2 + +[kube_node] +instance-3 + +[etcd] +instance-3 +instance-1 +instance-2 + +[broken_kube_control_plane] +instance-1 +instance-2 + +[broken_etcd] +instance-1 etcd_member_name=etcd2 +instance-2 etcd_member_name=etcd3 +{% endif %} + +[k8s_cluster:children] +kube_node +kube_control_plane +calico_rr + +[calico_rr] + +[fake_hosts] diff --git a/kubespray/tests/cloud_playbooks/roles/packet-ci/templates/vm.yml.j2 b/kubespray/tests/cloud_playbooks/roles/packet-ci/templates/vm.yml.j2 new file mode 100644 index 0000000..6a8e027 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/roles/packet-ci/templates/vm.yml.j2 @@ -0,0 +1,52 @@ +--- +apiVersion: kubevirt.io/v1alpha3 +kind: VirtualMachine +metadata: + name: "instance-{{ vm_id }}" + namespace: "{{ test_name }}" + labels: + kubevirt.io/os: {{ cloud_image }} +spec: + running: true + template: + metadata: + labels: + kubevirt.io/size: small + kubevirt.io/domain: "{{ test_name }}" + spec: + domain: + devices: + blockMultiQueue: true + disks: + - disk: + bus: virtio + name: containervolume + cache: writethrough + - disk: + bus: virtio + name: cloudinitvolume + interfaces: + - name: default + bridge: {} + cpu: + cores: {{ vm_cpu_cores }} + sockets: {{ vm_cpu_sockets }} + threads: {{ vm_cpu_threads }} + resources: + requests: + memory: {{ vm_memory * memory_allocation_ratio }} + cpu: {{ vm_cpu_cores * cpu_allocation_ratio }} + limits: + memory: {{ vm_memory }} + cpu: {{ vm_cpu_cores }} + networks: + - name: default + pod: {} + terminationGracePeriodSeconds: 0 + volumes: + - name: containervolume + containerDisk: + image: quay.io/kubespray/vm-{{ cloud_image }} + - name: cloudinitvolume + cloudInitNoCloud: + userDataBase64: {{ cloud_init[cloud_image] }} diff --git a/kubespray/tests/cloud_playbooks/templates/boto.j2 b/kubespray/tests/cloud_playbooks/templates/boto.j2 new file mode 100644 index 0000000..660f1a0 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/templates/boto.j2 @@ -0,0 +1,11 @@ +[Credentials] +gs_access_key_id = {{ gs_key }} +gs_secret_access_key = {{ gs_skey }} +[Boto] +https_validate_certificates = True +[GoogleCompute] +[GSUtil] +default_project_id = {{ gce_project_id }} +content_language = en +default_api_version = 2 +[OAuth2] diff --git a/kubespray/tests/cloud_playbooks/templates/gcs_life.json.j2 b/kubespray/tests/cloud_playbooks/templates/gcs_life.json.j2 new file mode 100644 index 0000000..a666c8f --- /dev/null +++ b/kubespray/tests/cloud_playbooks/templates/gcs_life.json.j2 @@ -0,0 +1,9 @@ +{ + "rule": + [ + { + "action": {"type": "Delete"}, + "condition": {"age": {{expire_days}}} + } + ] +} diff --git a/kubespray/tests/cloud_playbooks/upload-logs-gcs.yml b/kubespray/tests/cloud_playbooks/upload-logs-gcs.yml new file mode 100644 index 0000000..cae06f2 --- /dev/null +++ b/kubespray/tests/cloud_playbooks/upload-logs-gcs.yml @@ -0,0 +1,82 @@ +--- +- name: Upload logs to GCS + hosts: localhost + become: false + gather_facts: no + + vars: + expire_days: 2 + + tasks: + - name: Generate uniq bucket name prefix + raw: date +%Y%m%d + changed_when: false + register: out + + - name: Replace_test_id + set_fact: + test_name: "kargo-ci-{{ out.stdout_lines[0] }}" + + - name: Set file_name for logs + set_fact: + file_name: "{{ ostype }}-{{ kube_network_plugin }}-{{ commit }}-logs.tar.gz" + + - name: Create a bucket + community.google.gc_storage: + bucket: "{{ test_name }}" + mode: create + permission: public-read + gs_access_key: "{{ gs_key }}" + gs_secret_key: "{{ gs_skey }}" + no_log: True + + - name: Create a lifecycle template for the bucket + template: + src: gcs_life.json.j2 + dest: "{{ dir }}/gcs_life.json" + mode: 0644 + + - name: Create a boto config to access GCS + template: + src: boto.j2 + dest: "{{ dir }}/.boto" + mode: 0640 + no_log: True + + - name: Download gsutil cp installer + get_url: + url: https://dl.google.com/dl/cloudsdk/channels/rapid/install_google_cloud_sdk.bash + dest: "{{ dir }}/gcp-installer.sh" + mode: 0644 + + - name: Get gsutil tool + command: "{{ dir }}/gcp-installer.sh" + environment: + CLOUDSDK_CORE_DISABLE_PROMPTS: "1" + CLOUDSDK_INSTALL_DIR: "{{ dir }}" + no_log: True + failed_when: false + + - name: Apply the lifecycle rules + command: "{{ dir }}/google-cloud-sdk/bin/gsutil lifecycle set {{ dir }}/gcs_life.json gs://{{ test_name }}" + changed_when: false + environment: + BOTO_CONFIG: "{{ dir }}/.boto" + no_log: True + + - name: Upload collected diagnostic info + community.google.gc_storage: + bucket: "{{ test_name }}" + mode: put + permission: public-read + object: "{{ file_name }}" + src: "{{ dir }}/logs.tar.gz" + headers: '{"Content-Encoding": "x-gzip"}' + gs_access_key: "{{ gs_key }}" + gs_secret_key: "{{ gs_skey }}" + expiration: "{{ expire_days * 36000 | int }}" + failed_when: false + no_log: True + + - debug: # noqa name[missing] + msg: "A public url https://storage.googleapis.com/{{ test_name }}/{{ file_name }}" diff --git a/kubespray/tests/cloud_playbooks/wait-for-ssh.yml b/kubespray/tests/cloud_playbooks/wait-for-ssh.yml new file mode 100644 index 0000000..0e09c9f --- /dev/null +++ b/kubespray/tests/cloud_playbooks/wait-for-ssh.yml @@ -0,0 +1,13 @@ +--- +- name: Wait until SSH is available + hosts: all + become: False + gather_facts: False + + tasks: + - name: Wait until SSH is available + wait_for: + host: "{{ ansible_host }}" + port: 22 + timeout: 240 + delegate_to: localhost diff --git a/kubespray/tests/common/_docker_hub_registry_mirror.yml b/kubespray/tests/common/_docker_hub_registry_mirror.yml new file mode 100644 index 0000000..db521d6 --- /dev/null +++ b/kubespray/tests/common/_docker_hub_registry_mirror.yml @@ -0,0 +1,45 @@ +--- +docker_registry_mirrors: + - "https://mirror.gcr.io" + +containerd_grpc_max_recv_message_size: 16777216 +containerd_grpc_max_send_message_size: 16777216 + +containerd_registries: + "docker.io": + - "https://mirror.gcr.io" + - "https://registry-1.docker.io" + +containerd_registries_mirrors: + - prefix: docker.io + mirrors: + - host: https://mirror.gcr.io + capabilities: ["pull", "resolve"] + skip_verify: false + - host: https://registry-1.docker.io + capabilities: ["pull", "resolve"] + skip_verify: false + +containerd_max_container_log_line_size: -1 + +crio_registries: + - prefix: docker.io + insecure: false + blocked: false + unqualified: false + location: registry-1.docker.io + mirrors: + - location: mirror.gcr.io + insecure: false + +netcheck_agent_image_repo: "{{ quay_image_repo }}/kubespray/k8s-netchecker-agent" +netcheck_server_image_repo: "{{ quay_image_repo }}/kubespray/k8s-netchecker-server" + +nginx_image_repo: "{{ quay_image_repo }}/kubespray/nginx" + +flannel_image_repo: "{{ quay_image_repo }}/kubespray/flannel" +flannel_init_image_repo: "{{ quay_image_repo }}/kubespray/flannel-cni-plugin" + +# Kubespray settings for tests +deploy_netchecker: true +dns_min_replicas: 1 diff --git a/kubespray/tests/common/_kubespray_test_settings.yml b/kubespray/tests/common/_kubespray_test_settings.yml new file mode 100644 index 0000000..67da05c --- /dev/null +++ b/kubespray/tests/common/_kubespray_test_settings.yml @@ -0,0 +1,5 @@ +--- +# Kubespray settings for tests +deploy_netchecker: true +dns_min_replicas: 1 +unsafe_show_logs: true diff --git a/kubespray/tests/files/custom_cni/README.md b/kubespray/tests/files/custom_cni/README.md new file mode 100644 index 0000000..dd30240 --- /dev/null +++ b/kubespray/tests/files/custom_cni/README.md @@ -0,0 +1,11 @@ +# Custom CNI manifest generation + +As an example we are using Cilium for testing the network_plugins/custom_cni. + +To update the generated manifests to the latest version do the following: + +```sh +helm repo add cilium https://helm.cilium.io/ +helm repo update +helm template cilium/cilium -n kube-system -f values.yaml > cilium.yaml +``` diff --git a/kubespray/tests/files/custom_cni/cilium.yaml b/kubespray/tests/files/custom_cni/cilium.yaml new file mode 100644 index 0000000..9bd3bfb --- /dev/null +++ b/kubespray/tests/files/custom_cni/cilium.yaml @@ -0,0 +1,1056 @@ +--- +# Source: cilium/templates/cilium-agent/serviceaccount.yaml +apiVersion: v1 +kind: ServiceAccount +metadata: + name: "cilium" + namespace: kube-system +--- +# Source: cilium/templates/cilium-operator/serviceaccount.yaml +apiVersion: v1 +kind: ServiceAccount +metadata: + name: "cilium-operator" + namespace: kube-system +--- +# Source: cilium/templates/cilium-configmap.yaml +apiVersion: v1 +kind: ConfigMap +metadata: + name: cilium-config + namespace: kube-system +data: + + # Identity allocation mode selects how identities are shared between cilium + # nodes by setting how they are stored. The options are "crd" or "kvstore". + # - "crd" stores identities in kubernetes as CRDs (custom resource definition). + # These can be queried with: + # kubectl get ciliumid + # - "kvstore" stores identities in an etcd kvstore, that is + # configured below. Cilium versions before 1.6 supported only the kvstore + # backend. Upgrades from these older cilium versions should continue using + # the kvstore by commenting out the identity-allocation-mode below, or + # setting it to "kvstore". + identity-allocation-mode: crd + identity-heartbeat-timeout: "30m0s" + identity-gc-interval: "15m0s" + cilium-endpoint-gc-interval: "5m0s" + nodes-gc-interval: "5m0s" + skip-cnp-status-startup-clean: "false" + # Disable the usage of CiliumEndpoint CRD + disable-endpoint-crd: "false" + + # If you want to run cilium in debug mode change this value to true + debug: "false" + debug-verbose: "" + # The agent can be put into the following three policy enforcement modes + # default, always and never. + # https://docs.cilium.io/en/latest/security/policy/intro/#policy-enforcement-modes + enable-policy: "default" + + # Enable IPv4 addressing. If enabled, all endpoints are allocated an IPv4 + # address. + enable-ipv4: "true" + + # Enable IPv6 addressing. If enabled, all endpoints are allocated an IPv6 + # address. + enable-ipv6: "false" + # Users who wish to specify their own custom CNI configuration file must set + # custom-cni-conf to "true", otherwise Cilium may overwrite the configuration. + custom-cni-conf: "false" + enable-bpf-clock-probe: "true" + # If you want cilium monitor to aggregate tracing for packets, set this level + # to "low", "medium", or "maximum". The higher the level, the less packets + # that will be seen in monitor output. + monitor-aggregation: medium + + # The monitor aggregation interval governs the typical time between monitor + # notification events for each allowed connection. + # + # Only effective when monitor aggregation is set to "medium" or higher. + monitor-aggregation-interval: "5s" + + # The monitor aggregation flags determine which TCP flags which, upon the + # first observation, cause monitor notifications to be generated. + # + # Only effective when monitor aggregation is set to "medium" or higher. + monitor-aggregation-flags: all + # Specifies the ratio (0.0-1.0] of total system memory to use for dynamic + # sizing of the TCP CT, non-TCP CT, NAT and policy BPF maps. + bpf-map-dynamic-size-ratio: "0.0025" + # bpf-policy-map-max specifies the maximum number of entries in endpoint + # policy map (per endpoint) + bpf-policy-map-max: "16384" + # bpf-lb-map-max specifies the maximum number of entries in bpf lb service, + # backend and affinity maps. + bpf-lb-map-max: "65536" + bpf-lb-external-clusterip: "false" + + # Pre-allocation of map entries allows per-packet latency to be reduced, at + # the expense of up-front memory allocation for the entries in the maps. The + # default value below will minimize memory usage in the default installation; + # users who are sensitive to latency may consider setting this to "true". + # + # This option was introduced in Cilium 1.4. Cilium 1.3 and earlier ignore + # this option and behave as though it is set to "true". + # + # If this value is modified, then during the next Cilium startup the restore + # of existing endpoints and tracking of ongoing connections may be disrupted. + # As a result, reply packets may be dropped and the load-balancing decisions + # for established connections may change. + # + # If this option is set to "false" during an upgrade from 1.3 or earlier to + # 1.4 or later, then it may cause one-time disruptions during the upgrade. + preallocate-bpf-maps: "false" + + # Regular expression matching compatible Istio sidecar istio-proxy + # container image names + sidecar-istio-proxy-image: "cilium/istio_proxy" + + # Name of the cluster. Only relevant when building a mesh of clusters. + cluster-name: default + # Unique ID of the cluster. Must be unique across all conneted clusters and + # in the range of 1 and 255. Only relevant when building a mesh of clusters. + cluster-id: "0" + + # Encapsulation mode for communication between nodes + # Possible values: + # - disabled + # - vxlan (default) + # - geneve + tunnel: "vxlan" + + + # Enables L7 proxy for L7 policy enforcement and visibility + enable-l7-proxy: "true" + + enable-ipv4-masquerade: "true" + enable-ipv6-big-tcp: "false" + enable-ipv6-masquerade: "true" + + enable-xt-socket-fallback: "true" + install-iptables-rules: "true" + install-no-conntrack-iptables-rules: "false" + + auto-direct-node-routes: "false" + enable-local-redirect-policy: "false" + + kube-proxy-replacement: "disabled" + bpf-lb-sock: "false" + enable-health-check-nodeport: "true" + node-port-bind-protection: "true" + enable-auto-protect-node-port-range: "true" + enable-svc-source-range-check: "true" + enable-l2-neigh-discovery: "true" + arping-refresh-period: "30s" + enable-endpoint-health-checking: "true" + enable-health-checking: "true" + enable-well-known-identities: "false" + enable-remote-node-identity: "true" + synchronize-k8s-nodes: "true" + operator-api-serve-addr: "127.0.0.1:9234" + ipam: "cluster-pool" + cluster-pool-ipv4-cidr: "{{ kube_pods_subnet }}" + cluster-pool-ipv4-mask-size: "24" + disable-cnp-status-updates: "true" + enable-vtep: "false" + vtep-endpoint: "" + vtep-cidr: "" + vtep-mask: "" + vtep-mac: "" + enable-bgp-control-plane: "false" + procfs: "/host/proc" + bpf-root: "/sys/fs/bpf" + cgroup-root: "/run/cilium/cgroupv2" + enable-k8s-terminating-endpoint: "true" + enable-sctp: "false" + remove-cilium-node-taints: "true" + set-cilium-is-up-condition: "true" + unmanaged-pod-watcher-interval: "15" + tofqdns-dns-reject-response-code: "refused" + tofqdns-enable-dns-compression: "true" + tofqdns-endpoint-max-ip-per-hostname: "50" + tofqdns-idle-connection-grace-period: "0s" + tofqdns-max-deferred-connection-deletes: "10000" + tofqdns-min-ttl: "3600" + tofqdns-proxy-response-max-delay: "100ms" + agent-not-ready-taint-key: "node.cilium.io/agent-not-ready" +--- +# Source: cilium/templates/cilium-agent/clusterrole.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cilium + labels: + app.kubernetes.io/part-of: cilium +rules: +- apiGroups: + - networking.k8s.io + resources: + - networkpolicies + verbs: + - get + - list + - watch +- apiGroups: + - discovery.k8s.io + resources: + - endpointslices + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + - namespaces + - services + - pods + - endpoints + - nodes + verbs: + - get + - list + - watch +- apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + verbs: + - list + - watch + # This is used when validating policies in preflight. This will need to stay + # until we figure out how to avoid "get" inside the preflight, and then + # should be removed ideally. + - get +- apiGroups: + - cilium.io + resources: + - ciliumloadbalancerippools + - ciliumbgppeeringpolicies + - ciliumclusterwideenvoyconfigs + - ciliumclusterwidenetworkpolicies + - ciliumegressgatewaypolicies + - ciliumendpoints + - ciliumendpointslices + - ciliumenvoyconfigs + - ciliumidentities + - ciliumlocalredirectpolicies + - ciliumnetworkpolicies + - ciliumnodes + - ciliumnodeconfigs + verbs: + - list + - watch +- apiGroups: + - cilium.io + resources: + - ciliumidentities + - ciliumendpoints + - ciliumnodes + verbs: + - create +- apiGroups: + - cilium.io + # To synchronize garbage collection of such resources + resources: + - ciliumidentities + verbs: + - update +- apiGroups: + - cilium.io + resources: + - ciliumendpoints + verbs: + - delete + - get +- apiGroups: + - cilium.io + resources: + - ciliumnodes + - ciliumnodes/status + verbs: + - get + - update +- apiGroups: + - cilium.io + resources: + - ciliumnetworkpolicies/status + - ciliumclusterwidenetworkpolicies/status + - ciliumendpoints/status + - ciliumendpoints + verbs: + - patch +--- +# Source: cilium/templates/cilium-operator/clusterrole.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: cilium-operator + labels: + app.kubernetes.io/part-of: cilium +rules: +- apiGroups: + - "" + resources: + - pods + verbs: + - get + - list + - watch + # to automatically delete [core|kube]dns pods so that are starting to being + # managed by Cilium + - delete +- apiGroups: + - "" + resources: + - nodes + verbs: + - list + - watch +- apiGroups: + - "" + resources: + # To remove node taints + - nodes + # To set NetworkUnavailable false on startup + - nodes/status + verbs: + - patch +- apiGroups: + - discovery.k8s.io + resources: + - endpointslices + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + # to perform LB IP allocation for BGP + - services/status + verbs: + - update + - patch +- apiGroups: + - "" + resources: + # to check apiserver connectivity + - namespaces + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + # to perform the translation of a CNP that contains `ToGroup` to its endpoints + - services + - endpoints + verbs: + - get + - list + - watch +- apiGroups: + - cilium.io + resources: + - ciliumnetworkpolicies + - ciliumclusterwidenetworkpolicies + verbs: + # Create auto-generated CNPs and CCNPs from Policies that have 'toGroups' + - create + - update + - deletecollection + # To update the status of the CNPs and CCNPs + - patch + - get + - list + - watch +- apiGroups: + - cilium.io + resources: + - ciliumnetworkpolicies/status + - ciliumclusterwidenetworkpolicies/status + verbs: + # Update the auto-generated CNPs and CCNPs status. + - patch + - update +- apiGroups: + - cilium.io + resources: + - ciliumendpoints + - ciliumidentities + verbs: + # To perform garbage collection of such resources + - delete + - list + - watch +- apiGroups: + - cilium.io + resources: + - ciliumidentities + verbs: + # To synchronize garbage collection of such resources + - update +- apiGroups: + - cilium.io + resources: + - ciliumnodes + verbs: + - create + - update + - get + - list + - watch + # To perform CiliumNode garbage collector + - delete +- apiGroups: + - cilium.io + resources: + - ciliumnodes/status + verbs: + - update +- apiGroups: + - cilium.io + resources: + - ciliumendpointslices + - ciliumenvoyconfigs + verbs: + - create + - update + - get + - list + - watch + - delete + - patch +- apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + verbs: + - create + - get + - list + - watch +- apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + verbs: + - update + resourceNames: + - ciliumloadbalancerippools.cilium.io + - ciliumbgppeeringpolicies.cilium.io + - ciliumclusterwideenvoyconfigs.cilium.io + - ciliumclusterwidenetworkpolicies.cilium.io + - ciliumegressgatewaypolicies.cilium.io + - ciliumendpoints.cilium.io + - ciliumendpointslices.cilium.io + - ciliumenvoyconfigs.cilium.io + - ciliumexternalworkloads.cilium.io + - ciliumidentities.cilium.io + - ciliumlocalredirectpolicies.cilium.io + - ciliumnetworkpolicies.cilium.io + - ciliumnodes.cilium.io + - ciliumnodeconfigs.cilium.io +- apiGroups: + - cilium.io + resources: + - ciliumloadbalancerippools + verbs: + - get + - list + - watch +- apiGroups: + - cilium.io + resources: + - ciliumloadbalancerippools/status + verbs: + - patch +# For cilium-operator running in HA mode. +# +# Cilium operator running in HA mode requires the use of ResourceLock for Leader Election +# between multiple running instances. +# The preferred way of doing this is to use LeasesResourceLock as edits to Leases are less +# common and fewer objects in the cluster watch "all Leases". +- apiGroups: + - coordination.k8s.io + resources: + - leases + verbs: + - create + - get + - update +--- +# Source: cilium/templates/cilium-agent/clusterrolebinding.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cilium + labels: + app.kubernetes.io/part-of: cilium +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cilium +subjects: +- kind: ServiceAccount + name: "cilium" + namespace: kube-system +--- +# Source: cilium/templates/cilium-operator/clusterrolebinding.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: cilium-operator + labels: + app.kubernetes.io/part-of: cilium +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cilium-operator +subjects: +- kind: ServiceAccount + name: "cilium-operator" + namespace: kube-system +--- +# Source: cilium/templates/cilium-agent/role.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: cilium-config-agent + namespace: kube-system + labels: + app.kubernetes.io/part-of: cilium +rules: +- apiGroups: + - "" + resources: + - configmaps + verbs: + - get + - list + - watch +--- +# Source: cilium/templates/cilium-agent/rolebinding.yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: cilium-config-agent + namespace: kube-system + labels: + app.kubernetes.io/part-of: cilium +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: cilium-config-agent +subjects: + - kind: ServiceAccount + name: "cilium" + namespace: kube-system +--- +# Source: cilium/templates/cilium-agent/daemonset.yaml +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: cilium + namespace: kube-system + labels: + k8s-app: cilium + app.kubernetes.io/part-of: cilium + app.kubernetes.io/name: cilium-agent +spec: + selector: + matchLabels: + k8s-app: cilium + updateStrategy: + rollingUpdate: + maxUnavailable: 2 + type: RollingUpdate + template: + metadata: + annotations: + # Set app AppArmor's profile to "unconfined". The value of this annotation + # can be modified as long users know which profiles they have available + # in AppArmor. + container.apparmor.security.beta.kubernetes.io/cilium-agent: "unconfined" + container.apparmor.security.beta.kubernetes.io/clean-cilium-state: "unconfined" + container.apparmor.security.beta.kubernetes.io/mount-cgroup: "unconfined" + container.apparmor.security.beta.kubernetes.io/apply-sysctl-overwrites: "unconfined" + labels: + k8s-app: cilium + app.kubernetes.io/name: cilium-agent + app.kubernetes.io/part-of: cilium + spec: + containers: + - name: cilium-agent + image: "quay.io/cilium/cilium:v1.13.0@sha256:6544a3441b086a2e09005d3e21d1a4afb216fae19c5a60b35793c8a9438f8f68" + imagePullPolicy: IfNotPresent + command: + - cilium-agent + args: + - --config-dir=/tmp/cilium/config-map + startupProbe: + httpGet: + host: "127.0.0.1" + path: /healthz + port: 9879 + scheme: HTTP + httpHeaders: + - name: "brief" + value: "true" + failureThreshold: 105 + periodSeconds: 2 + successThreshold: 1 + livenessProbe: + httpGet: + host: "127.0.0.1" + path: /healthz + port: 9879 + scheme: HTTP + httpHeaders: + - name: "brief" + value: "true" + periodSeconds: 30 + successThreshold: 1 + failureThreshold: 10 + timeoutSeconds: 5 + readinessProbe: + httpGet: + host: "127.0.0.1" + path: /healthz + port: 9879 + scheme: HTTP + httpHeaders: + - name: "brief" + value: "true" + periodSeconds: 30 + successThreshold: 1 + failureThreshold: 3 + timeoutSeconds: 5 + env: + - name: K8S_NODE_NAME + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: spec.nodeName + - name: CILIUM_K8S_NAMESPACE + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + - name: CILIUM_CLUSTERMESH_CONFIG + value: /var/lib/cilium/clustermesh/ + - name: CILIUM_CNI_CHAINING_MODE + valueFrom: + configMapKeyRef: + name: cilium-config + key: cni-chaining-mode + optional: true + - name: CILIUM_CUSTOM_CNI_CONF + valueFrom: + configMapKeyRef: + name: cilium-config + key: custom-cni-conf + optional: true + lifecycle: + postStart: + exec: + command: + - "/cni-install.sh" + - "--enable-debug=false" + - "--cni-exclusive=true" + - "--log-file=/var/run/cilium/cilium-cni.log" + preStop: + exec: + command: + - /cni-uninstall.sh + securityContext: + seLinuxOptions: + level: s0 + type: spc_t + capabilities: + add: + - CHOWN + - KILL + - NET_ADMIN + - NET_RAW + - IPC_LOCK + - SYS_MODULE + - SYS_ADMIN + - SYS_RESOURCE + - DAC_OVERRIDE + - FOWNER + - SETGID + - SETUID + drop: + - ALL + terminationMessagePolicy: FallbackToLogsOnError + volumeMounts: + # Unprivileged containers need to mount /proc/sys/net from the host + # to have write access + - mountPath: /host/proc/sys/net + name: host-proc-sys-net + # Unprivileged containers need to mount /proc/sys/kernel from the host + # to have write access + - mountPath: /host/proc/sys/kernel + name: host-proc-sys-kernel + - name: bpf-maps + mountPath: /sys/fs/bpf + # Unprivileged containers can't set mount propagation to bidirectional + # in this case we will mount the bpf fs from an init container that + # is privileged and set the mount propagation from host to container + # in Cilium. + mountPropagation: HostToContainer + - name: cilium-run + mountPath: /var/run/cilium + - name: cni-path + mountPath: /host/opt/cni/bin + - name: etc-cni-netd + mountPath: /host/etc/cni/net.d + - name: clustermesh-secrets + mountPath: /var/lib/cilium/clustermesh + readOnly: true + # Needed to be able to load kernel modules + - name: lib-modules + mountPath: /lib/modules + readOnly: true + - name: xtables-lock + mountPath: /run/xtables.lock + - name: tmp + mountPath: /tmp + initContainers: + - name: config + image: "quay.io/cilium/cilium:v1.13.0@sha256:6544a3441b086a2e09005d3e21d1a4afb216fae19c5a60b35793c8a9438f8f68" + imagePullPolicy: IfNotPresent + command: + - cilium + - build-config + env: + - name: K8S_NODE_NAME + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: spec.nodeName + - name: CILIUM_K8S_NAMESPACE + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + volumeMounts: + - name: tmp + mountPath: /tmp + terminationMessagePolicy: FallbackToLogsOnError + # Required to mount cgroup2 filesystem on the underlying Kubernetes node. + # We use nsenter command with host's cgroup and mount namespaces enabled. + - name: mount-cgroup + image: "quay.io/cilium/cilium:v1.13.0@sha256:6544a3441b086a2e09005d3e21d1a4afb216fae19c5a60b35793c8a9438f8f68" + imagePullPolicy: IfNotPresent + env: + - name: CGROUP_ROOT + value: /run/cilium/cgroupv2 + - name: BIN_PATH + value: /opt/cni/bin + command: + - sh + - -ec + # The statically linked Go program binary is invoked to avoid any + # dependency on utilities like sh and mount that can be missing on certain + # distros installed on the underlying host. Copy the binary to the + # same directory where we install cilium cni plugin so that exec permissions + # are available. + - | + cp /usr/bin/cilium-mount /hostbin/cilium-mount; + nsenter --cgroup=/hostproc/1/ns/cgroup --mount=/hostproc/1/ns/mnt "${BIN_PATH}/cilium-mount" $CGROUP_ROOT; + rm /hostbin/cilium-mount + volumeMounts: + - name: hostproc + mountPath: /hostproc + - name: cni-path + mountPath: /hostbin + terminationMessagePolicy: FallbackToLogsOnError + securityContext: + seLinuxOptions: + level: s0 + type: spc_t + capabilities: + add: + - SYS_ADMIN + - SYS_CHROOT + - SYS_PTRACE + drop: + - ALL + - name: apply-sysctl-overwrites + image: "quay.io/cilium/cilium:v1.13.0@sha256:6544a3441b086a2e09005d3e21d1a4afb216fae19c5a60b35793c8a9438f8f68" + imagePullPolicy: IfNotPresent + env: + - name: BIN_PATH + value: /opt/cni/bin + command: + - sh + - -ec + # The statically linked Go program binary is invoked to avoid any + # dependency on utilities like sh that can be missing on certain + # distros installed on the underlying host. Copy the binary to the + # same directory where we install cilium cni plugin so that exec permissions + # are available. + - | + cp /usr/bin/cilium-sysctlfix /hostbin/cilium-sysctlfix; + nsenter --mount=/hostproc/1/ns/mnt "${BIN_PATH}/cilium-sysctlfix"; + rm /hostbin/cilium-sysctlfix + volumeMounts: + - name: hostproc + mountPath: /hostproc + - name: cni-path + mountPath: /hostbin + terminationMessagePolicy: FallbackToLogsOnError + securityContext: + seLinuxOptions: + level: s0 + type: spc_t + capabilities: + add: + - SYS_ADMIN + - SYS_CHROOT + - SYS_PTRACE + drop: + - ALL + # Mount the bpf fs if it is not mounted. We will perform this task + # from a privileged container because the mount propagation bidirectional + # only works from privileged containers. + - name: mount-bpf-fs + image: "quay.io/cilium/cilium:v1.13.0@sha256:6544a3441b086a2e09005d3e21d1a4afb216fae19c5a60b35793c8a9438f8f68" + imagePullPolicy: IfNotPresent + args: + - 'mount | grep "/sys/fs/bpf type bpf" || mount -t bpf bpf /sys/fs/bpf' + command: + - /bin/bash + - -c + - -- + terminationMessagePolicy: FallbackToLogsOnError + securityContext: + privileged: true + volumeMounts: + - name: bpf-maps + mountPath: /sys/fs/bpf + mountPropagation: Bidirectional + - name: clean-cilium-state + image: "quay.io/cilium/cilium:v1.13.0@sha256:6544a3441b086a2e09005d3e21d1a4afb216fae19c5a60b35793c8a9438f8f68" + imagePullPolicy: IfNotPresent + command: + - /init-container.sh + env: + - name: CILIUM_ALL_STATE + valueFrom: + configMapKeyRef: + name: cilium-config + key: clean-cilium-state + optional: true + - name: CILIUM_BPF_STATE + valueFrom: + configMapKeyRef: + name: cilium-config + key: clean-cilium-bpf-state + optional: true + terminationMessagePolicy: FallbackToLogsOnError + securityContext: + seLinuxOptions: + level: s0 + type: spc_t + capabilities: + add: + - NET_ADMIN + - SYS_MODULE + - SYS_ADMIN + - SYS_RESOURCE + drop: + - ALL + volumeMounts: + - name: bpf-maps + mountPath: /sys/fs/bpf + # Required to mount cgroup filesystem from the host to cilium agent pod + - name: cilium-cgroup + mountPath: /run/cilium/cgroupv2 + mountPropagation: HostToContainer + - name: cilium-run + mountPath: /var/run/cilium + resources: + requests: + cpu: 100m + memory: 100Mi # wait-for-kube-proxy + restartPolicy: Always + priorityClassName: system-node-critical + serviceAccount: "cilium" + serviceAccountName: "cilium" + terminationGracePeriodSeconds: 1 + hostNetwork: true + affinity: + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchLabels: + k8s-app: cilium + topologyKey: kubernetes.io/hostname + nodeSelector: + kubernetes.io/os: linux + tolerations: + - operator: Exists + volumes: + # For sharing configuration between the "config" initContainer and the agent + - name: tmp + emptyDir: {} + # To keep state between restarts / upgrades + - name: cilium-run + hostPath: + path: /var/run/cilium + type: DirectoryOrCreate + # To keep state between restarts / upgrades for bpf maps + - name: bpf-maps + hostPath: + path: /sys/fs/bpf + type: DirectoryOrCreate + # To mount cgroup2 filesystem on the host + - name: hostproc + hostPath: + path: /proc + type: Directory + # To keep state between restarts / upgrades for cgroup2 filesystem + - name: cilium-cgroup + hostPath: + path: /run/cilium/cgroupv2 + type: DirectoryOrCreate + # To install cilium cni plugin in the host + - name: cni-path + hostPath: + path: /opt/cni/bin + type: DirectoryOrCreate + # To install cilium cni configuration in the host + - name: etc-cni-netd + hostPath: + path: /etc/cni/net.d + type: DirectoryOrCreate + # To be able to load kernel modules + - name: lib-modules + hostPath: + path: /lib/modules + # To access iptables concurrently with other processes (e.g. kube-proxy) + - name: xtables-lock + hostPath: + path: /run/xtables.lock + type: FileOrCreate + # To read the clustermesh configuration + - name: clustermesh-secrets + secret: + secretName: cilium-clustermesh + # note: the leading zero means this number is in octal representation: do not remove it + defaultMode: 0400 + optional: true + - name: host-proc-sys-net + hostPath: + path: /proc/sys/net + type: Directory + - name: host-proc-sys-kernel + hostPath: + path: /proc/sys/kernel + type: Directory +--- +# Source: cilium/templates/cilium-operator/deployment.yaml +apiVersion: apps/v1 +kind: Deployment +metadata: + name: cilium-operator + namespace: kube-system + labels: + io.cilium/app: operator + name: cilium-operator + app.kubernetes.io/part-of: cilium + app.kubernetes.io/name: cilium-operator +spec: + # See docs on ServerCapabilities.LeasesResourceLock in file pkg/k8s/version/version.go + # for more details. + replicas: 2 + selector: + matchLabels: + io.cilium/app: operator + name: cilium-operator + strategy: + rollingUpdate: + maxSurge: 1 + maxUnavailable: 1 + type: RollingUpdate + template: + metadata: + annotations: + labels: + io.cilium/app: operator + name: cilium-operator + app.kubernetes.io/part-of: cilium + app.kubernetes.io/name: cilium-operator + spec: + containers: + - name: cilium-operator + image: "quay.io/cilium/operator-generic:v1.13.0@sha256:4b58d5b33e53378355f6e8ceb525ccf938b7b6f5384b35373f1f46787467ebf5" + imagePullPolicy: IfNotPresent + command: + - cilium-operator-generic + args: + - --config-dir=/tmp/cilium/config-map + - --debug=$(CILIUM_DEBUG) + env: + - name: K8S_NODE_NAME + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: spec.nodeName + - name: CILIUM_K8S_NAMESPACE + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + - name: CILIUM_DEBUG + valueFrom: + configMapKeyRef: + key: debug + name: cilium-config + optional: true + livenessProbe: + httpGet: + host: "127.0.0.1" + path: /healthz + port: 9234 + scheme: HTTP + initialDelaySeconds: 60 + periodSeconds: 10 + timeoutSeconds: 3 + volumeMounts: + - name: cilium-config-path + mountPath: /tmp/cilium/config-map + readOnly: true + terminationMessagePolicy: FallbackToLogsOnError + hostNetwork: true + restartPolicy: Always + priorityClassName: system-cluster-critical + serviceAccount: "cilium-operator" + serviceAccountName: "cilium-operator" + # In HA mode, cilium-operator pods must not be scheduled on the same + # node as they will clash with each other. + affinity: + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchLabels: + io.cilium/app: operator + topologyKey: kubernetes.io/hostname + nodeSelector: + kubernetes.io/os: linux + tolerations: + - operator: Exists + volumes: + # To read the configuration from the config map + - name: cilium-config-path + configMap: + name: cilium-config +--- +# Source: cilium/templates/cilium-secrets-namespace.yaml +# Only create the namespace if it's different from Ingress secret namespace or Ingress is not enabled. diff --git a/kubespray/tests/files/custom_cni/values.yaml b/kubespray/tests/files/custom_cni/values.yaml new file mode 100644 index 0000000..bba8cf7 --- /dev/null +++ b/kubespray/tests/files/custom_cni/values.yaml @@ -0,0 +1,11 @@ +--- + +# We disable hubble so that helm doesn't try to generate any certificate. +# This is not needed to test network_plugin/custom_cni anyway. +hubble: + enabled: false + +ipam: + operator: + # Set the appropriate pods subnet + clusterPoolIPv4PodCIDR: "{{ kube_pods_subnet }}" diff --git a/kubespray/tests/files/packet_almalinux8-calico-ha-ebpf.yml b/kubespray/tests/files/packet_almalinux8-calico-ha-ebpf.yml new file mode 100644 index 0000000..e4f4bb6 --- /dev/null +++ b/kubespray/tests/files/packet_almalinux8-calico-ha-ebpf.yml @@ -0,0 +1,10 @@ +--- +# Instance settings +cloud_image: almalinux-8 +mode: ha +vm_memory: 3072Mi + +# Kubespray settings +calico_bpf_enabled: true +loadbalancer_apiserver_localhost: true +auto_renew_certificates: true diff --git a/kubespray/tests/files/packet_almalinux8-calico-nodelocaldns-secondary.yml b/kubespray/tests/files/packet_almalinux8-calico-nodelocaldns-secondary.yml new file mode 100644 index 0000000..52ef869 --- /dev/null +++ b/kubespray/tests/files/packet_almalinux8-calico-nodelocaldns-secondary.yml @@ -0,0 +1,9 @@ +--- +# Instance settings +cloud_image: almalinux-8 +mode: default +vm_memory: 3072Mi + +# Kubespray settings +enable_nodelocaldns_secondary: true +loadbalancer_apiserver_type: haproxy diff --git a/kubespray/tests/files/packet_almalinux8-calico-remove-node.yml b/kubespray/tests/files/packet_almalinux8-calico-remove-node.yml new file mode 100644 index 0000000..4cb5dfc --- /dev/null +++ b/kubespray/tests/files/packet_almalinux8-calico-remove-node.yml @@ -0,0 +1,7 @@ +--- +# Instance settings +cloud_image: almalinux-8 +mode: ha + +# Kubespray settings +auto_renew_certificates: true diff --git a/kubespray/tests/files/packet_almalinux8-calico.yml b/kubespray/tests/files/packet_almalinux8-calico.yml new file mode 100644 index 0000000..63cf8bf --- /dev/null +++ b/kubespray/tests/files/packet_almalinux8-calico.yml @@ -0,0 +1,19 @@ +--- +# Instance settings +cloud_image: almalinux-8 +mode: default +vm_memory: 3072Mi + +# Kubespray settings +metrics_server_enabled: true +dashboard_namespace: "kube-dashboard" +dashboard_enabled: true +loadbalancer_apiserver_type: haproxy +local_path_provisioner_enabled: true + +# NTP mangement +ntp_enabled: true +ntp_timezone: Etc/UTC +ntp_manage_config: true +ntp_tinker_panic: true +ntp_force_sync_immediately: true diff --git a/kubespray/tests/files/packet_almalinux8-crio.yml b/kubespray/tests/files/packet_almalinux8-crio.yml new file mode 100644 index 0000000..35fa009 --- /dev/null +++ b/kubespray/tests/files/packet_almalinux8-crio.yml @@ -0,0 +1,8 @@ +--- +# Instance settings +cloud_image: almalinux-8 +mode: default + +# Kubespray settings +container_manager: crio +auto_renew_certificates: true diff --git a/kubespray/tests/files/packet_almalinux8-docker.yml b/kubespray/tests/files/packet_almalinux8-docker.yml new file mode 100644 index 0000000..bcc69cd --- /dev/null +++ b/kubespray/tests/files/packet_almalinux8-docker.yml @@ -0,0 +1,10 @@ +--- +# Instance settings +cloud_image: almalinux-8 +mode: default +vm_memory: 3072Mi + +# Use docker +container_manager: docker +etcd_deployment_type: docker +resolvconf_mode: docker_dns diff --git a/kubespray/tests/files/packet_almalinux8-kube-ovn.yml b/kubespray/tests/files/packet_almalinux8-kube-ovn.yml new file mode 100644 index 0000000..15dbabb --- /dev/null +++ b/kubespray/tests/files/packet_almalinux8-kube-ovn.yml @@ -0,0 +1,8 @@ +--- +# Instance settings +cloud_image: almalinux-8 +mode: default +vm_memory: 3072Mi + +# Kubespray settings +kube_network_plugin: kube-ovn diff --git a/kubespray/tests/files/packet_amazon-linux-2-aio.yml b/kubespray/tests/files/packet_amazon-linux-2-aio.yml new file mode 100644 index 0000000..7b2c69b --- /dev/null +++ b/kubespray/tests/files/packet_amazon-linux-2-aio.yml @@ -0,0 +1,4 @@ +--- +# Instance settings +cloud_image: amazon-linux-2 +mode: aio diff --git a/kubespray/tests/files/packet_centos7-calico-ha-once-localhost.yml b/kubespray/tests/files/packet_centos7-calico-ha-once-localhost.yml new file mode 100644 index 0000000..950aae0 --- /dev/null +++ b/kubespray/tests/files/packet_centos7-calico-ha-once-localhost.yml @@ -0,0 +1,18 @@ +--- +# Instance settings +cloud_image: centos-7 +mode: ha + +# Kubespray settings +download_localhost: true +download_run_once: true +typha_enabled: true +calico_apiserver_enabled: true +calico_backend: kdd +typha_secure: true +disable_ipv6_dns: true +auto_renew_certificates: true + +# Docker settings +container_manager: docker +etcd_deployment_type: docker diff --git a/kubespray/tests/files/packet_centos7-calico-ha.yml b/kubespray/tests/files/packet_centos7-calico-ha.yml new file mode 100644 index 0000000..be93a60 --- /dev/null +++ b/kubespray/tests/files/packet_centos7-calico-ha.yml @@ -0,0 +1,13 @@ +--- +# Instance settings +cloud_image: centos-7 +mode: ha + +# Kubespray settings +download_localhost: false +download_run_once: true +typha_enabled: true +calico_apiserver_enabled: true +calico_backend: kdd +typha_secure: true +auto_renew_certificates: true diff --git a/kubespray/tests/files/packet_centos7-flannel-addons-ha.yml b/kubespray/tests/files/packet_centos7-flannel-addons-ha.yml new file mode 100644 index 0000000..ef7cb1b --- /dev/null +++ b/kubespray/tests/files/packet_centos7-flannel-addons-ha.yml @@ -0,0 +1,74 @@ +--- +# Instance settings +cloud_image: centos-7 +mode: ha + +# Kubespray settings +kubeadm_certificate_key: 3998c58db6497dd17d909394e62d515368c06ec617710d02edea31c06d741085 +kube_proxy_mode: iptables +kube_network_plugin: flannel +download_localhost: false +download_run_once: true +helm_enabled: true +krew_enabled: true +kubernetes_audit: true +etcd_events_cluster_enabled: true +local_volume_provisioner_enabled: true +kube_encrypt_secret_data: true +ingress_nginx_enabled: true +ingress_nginx_webhook_enabled: true +ingress_nginx_webhook_job_ttl: 30 +cert_manager_enabled: true +# Disable as health checks are still unstable and slow to respond. +metrics_server_enabled: false +metrics_server_kubelet_insecure_tls: true +kube_token_auth: true +enable_nodelocaldns: false +kubelet_rotate_server_certificates: true +kubelet_csr_approver_enabled: false + +kube_oidc_url: https://accounts.google.com/.well-known/openid-configuration +kube_oidc_client_id: kubespray-example + +tls_min_version: "VersionTLS12" +tls_cipher_suites: + - TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 + +# test etcd tls cipher suites +etcd_tls_cipher_suites: + - TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 + - TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 + +# Containerd +containerd_storage_dir: /var/data/containerd +containerd_state_dir: /run/cri/containerd +containerd_oom_score: -999 + +# Kube-vip +kube_vip_enabled: true +kube_vip_arp_enabled: true +kube_vip_controlplane_enabled: true +kube_vip_address: 192.168.1.100 + +# MetalLB +metallb_enabled: true +metallb_speaker_enabled: true +metallb_config: + address_pools: + primary: + ip_range: + - 192.0.1.0-192.0.1.254 + auto_assign: true + pool1: + ip_range: + - 192.0.2.1-192.0.2.1 + auto_assign: false + pool2: + ip_range: + - 192.0.2.2-192.0.2.2 + auto_assign: false + + layer2: + - primary + - pool1 + - pool2 diff --git a/kubespray/tests/files/packet_centos7-multus-calico.yml b/kubespray/tests/files/packet_centos7-multus-calico.yml new file mode 100644 index 0000000..350c101 --- /dev/null +++ b/kubespray/tests/files/packet_centos7-multus-calico.yml @@ -0,0 +1,7 @@ +--- +# Instance settings +cloud_image: centos-7 +mode: default + +# Kubespray settings +kube_network_plugin_multus: true diff --git a/kubespray/tests/files/packet_centos7-weave-upgrade-ha.yml b/kubespray/tests/files/packet_centos7-weave-upgrade-ha.yml new file mode 100644 index 0000000..e290ae4 --- /dev/null +++ b/kubespray/tests/files/packet_centos7-weave-upgrade-ha.yml @@ -0,0 +1,11 @@ +--- +# Instance settings +cloud_image: centos-7 +mode: ha + +# Kubespray settings +kube_network_plugin: weave +kubernetes_audit: true + +# Needed to upgrade from 1.16 to 1.17, otherwise upgrade is partial and bug followed +upgrade_cluster_setup: true diff --git a/kubespray/tests/files/packet_debian10-calico.yml b/kubespray/tests/files/packet_debian10-calico.yml new file mode 100644 index 0000000..90e982a --- /dev/null +++ b/kubespray/tests/files/packet_debian10-calico.yml @@ -0,0 +1,11 @@ +--- +# Instance settings +cloud_image: debian-10 +mode: default + +# Kubespray settings +auto_renew_certificates: true + +# plugins +helm_enabled: true +krew_enabled: true diff --git a/kubespray/tests/files/packet_debian10-cilium-svc-proxy.yml b/kubespray/tests/files/packet_debian10-cilium-svc-proxy.yml new file mode 100644 index 0000000..3dcbc7a --- /dev/null +++ b/kubespray/tests/files/packet_debian10-cilium-svc-proxy.yml @@ -0,0 +1,10 @@ +--- +# Instance settings +cloud_image: debian-10 +mode: ha + +# Kubespray settings +kube_network_plugin: cilium +enable_network_policy: true + +cilium_kube_proxy_replacement: strict diff --git a/kubespray/tests/files/packet_debian10-docker.yml b/kubespray/tests/files/packet_debian10-docker.yml new file mode 100644 index 0000000..fc55e7f --- /dev/null +++ b/kubespray/tests/files/packet_debian10-docker.yml @@ -0,0 +1,9 @@ +--- +# Instance settings +cloud_image: debian-10 +mode: default + +# Use docker +container_manager: docker +etcd_deployment_type: docker +resolvconf_mode: docker_dns diff --git a/kubespray/tests/files/packet_debian10-macvlan.yml b/kubespray/tests/files/packet_debian10-macvlan.yml new file mode 100644 index 0000000..fad162e --- /dev/null +++ b/kubespray/tests/files/packet_debian10-macvlan.yml @@ -0,0 +1,11 @@ +--- +# Instance settings +cloud_image: debian-10 +mode: default + +# Kubespray settings +kube_network_plugin: macvlan +enable_nodelocaldns: false +kube_proxy_masquerade_all: true +macvlan_interface: "eth0" +auto_renew_certificates: true diff --git a/kubespray/tests/files/packet_debian11-calico-upgrade-once.yml b/kubespray/tests/files/packet_debian11-calico-upgrade-once.yml new file mode 100644 index 0000000..3c589a0 --- /dev/null +++ b/kubespray/tests/files/packet_debian11-calico-upgrade-once.yml @@ -0,0 +1,16 @@ +--- +# Instance settings +cloud_image: debian-11 +mode: default + +# Kubespray settings +download_run_once: true + +# Pin disabling ipip mode to ensure proper upgrade +ipip: false +calico_pool_blocksize: 26 +calico_vxlan_mode: Always +calico_network_backend: bird + +# Needed to bypass deprecation check +ignore_assert_errors: true diff --git a/kubespray/tests/files/packet_debian11-calico-upgrade.yml b/kubespray/tests/files/packet_debian11-calico-upgrade.yml new file mode 100644 index 0000000..1b05714 --- /dev/null +++ b/kubespray/tests/files/packet_debian11-calico-upgrade.yml @@ -0,0 +1,13 @@ +--- +# Instance settings +cloud_image: debian-11 +mode: default + +# Pin disabling ipip mode to ensure proper upgrade +ipip: false +calico_pool_blocksize: 26 +calico_vxlan_mode: Always +calico_network_backend: bird + +# Needed to bypass deprecation check +ignore_assert_errors: true diff --git a/kubespray/tests/files/packet_debian11-calico.yml b/kubespray/tests/files/packet_debian11-calico.yml new file mode 100644 index 0000000..61b31c2 --- /dev/null +++ b/kubespray/tests/files/packet_debian11-calico.yml @@ -0,0 +1,4 @@ +--- +# Instance settings +cloud_image: debian-11 +mode: default diff --git a/kubespray/tests/files/packet_debian11-custom-cni.yml b/kubespray/tests/files/packet_debian11-custom-cni.yml new file mode 100644 index 0000000..407423e --- /dev/null +++ b/kubespray/tests/files/packet_debian11-custom-cni.yml @@ -0,0 +1,9 @@ +--- +# Instance settings +cloud_image: debian-11 +mode: default + +# Kubespray settings +kube_network_plugin: custom_cni +custom_cni_manifests: + - "{{ playbook_dir }}/../tests/files/custom_cni/cilium.yaml" diff --git a/kubespray/tests/files/packet_debian11-docker.yml b/kubespray/tests/files/packet_debian11-docker.yml new file mode 100644 index 0000000..69ec8eb --- /dev/null +++ b/kubespray/tests/files/packet_debian11-docker.yml @@ -0,0 +1,9 @@ +--- +# Instance settings +cloud_image: debian-11 +mode: default + +# Use docker +container_manager: docker +etcd_deployment_type: docker +resolvconf_mode: docker_dns diff --git a/kubespray/tests/files/packet_debian11-kubelet-csr-approver.yml b/kubespray/tests/files/packet_debian11-kubelet-csr-approver.yml new file mode 100644 index 0000000..d1be098 --- /dev/null +++ b/kubespray/tests/files/packet_debian11-kubelet-csr-approver.yml @@ -0,0 +1,11 @@ +--- +# Instance settings +cloud_image: debian-11 +mode: default + +# Kubespray settings +kubelet_rotate_server_certificates: true +kubelet_csr_approver_enabled: true +kubelet_csr_approver_values: + # Do not check DNS resolution in testing (not recommended in production) + bypassDnsResolution: true diff --git a/kubespray/tests/files/packet_debian12-calico.yml b/kubespray/tests/files/packet_debian12-calico.yml new file mode 100644 index 0000000..a4adafc --- /dev/null +++ b/kubespray/tests/files/packet_debian12-calico.yml @@ -0,0 +1,4 @@ +--- +# Instance settings +cloud_image: debian-12 +mode: default diff --git a/kubespray/tests/files/packet_debian12-cilium.yml b/kubespray/tests/files/packet_debian12-cilium.yml new file mode 100644 index 0000000..c77bcf3 --- /dev/null +++ b/kubespray/tests/files/packet_debian12-cilium.yml @@ -0,0 +1,7 @@ +--- +# Instance settings +cloud_image: debian-12 +mode: default + +# Kubespray settings +kube_network_plugin: cilium diff --git a/kubespray/tests/files/packet_debian12-docker.yml b/kubespray/tests/files/packet_debian12-docker.yml new file mode 100644 index 0000000..5d4ac53 --- /dev/null +++ b/kubespray/tests/files/packet_debian12-docker.yml @@ -0,0 +1,9 @@ +--- +# Instance settings +cloud_image: debian-12 +mode: default + +# Use docker +container_manager: docker +etcd_deployment_type: docker +resolvconf_mode: docker_dns diff --git a/kubespray/tests/files/packet_fedora37-calico-selinux.yml b/kubespray/tests/files/packet_fedora37-calico-selinux.yml new file mode 100644 index 0000000..2fbbd7b --- /dev/null +++ b/kubespray/tests/files/packet_fedora37-calico-selinux.yml @@ -0,0 +1,14 @@ +--- +# Instance settings +cloud_image: fedora-37 +mode: default + +# Kubespray settings +auto_renew_certificates: true +# Switching to iptable due to https://github.com/projectcalico/calico/issues/5011 +# Kubernetes v1.23.0 kube-proxy does use v.7.x now. Calico v3.20.x/v3.21.x Pods show the following error +# Bad return code from 'ipset list'. error=exit status 1 family="inet" stderr="ipset v7.1: Kernel and userspace incompatible: settype hash:ip,port with revision 6 not supported by userspace. +kube_proxy_mode: iptables + +# Test with SELinux in enforcing mode +preinstall_selinux_state: enforcing diff --git a/kubespray/tests/files/packet_fedora37-calico-swap-selinux.yml b/kubespray/tests/files/packet_fedora37-calico-swap-selinux.yml new file mode 100644 index 0000000..775c930 --- /dev/null +++ b/kubespray/tests/files/packet_fedora37-calico-swap-selinux.yml @@ -0,0 +1,19 @@ +--- +# Instance settings +cloud_image: fedora-37 +mode: default + +# Kubespray settings +auto_renew_certificates: true +# Switching to iptable due to https://github.com/projectcalico/calico/issues/5011 +# Kubernetes v1.23.0 kube-proxy does use v.7.x now. Calico v3.20.x/v3.21.x Pods show the following error +# Bad return code from 'ipset list'. error=exit status 1 family="inet" stderr="ipset v7.1: Kernel and userspace incompatible: settype hash:ip,port with revision 6 not supported by userspace. +kube_proxy_mode: iptables + +# Test with SELinux in enforcing mode +preinstall_selinux_state: enforcing + +# Test Alpha swap feature by leveraging zswap default config in Fedora 35 +kubelet_fail_swap_on: false +kube_feature_gates: + - "NodeSwap=True" diff --git a/kubespray/tests/files/packet_fedora37-crio.yml b/kubespray/tests/files/packet_fedora37-crio.yml new file mode 100644 index 0000000..5f5e736 --- /dev/null +++ b/kubespray/tests/files/packet_fedora37-crio.yml @@ -0,0 +1,15 @@ +--- +# Instance settings +cloud_image: fedora-37 +mode: default + +# Kubespray settings +container_manager: crio +auto_renew_certificates: true +# Switching to iptable due to https://github.com/projectcalico/calico/issues/5011 +# Kubernetes v1.23.0 kube-proxy does use v.7.x now. Calico v3.20.x/v3.21.x Pods show the following error +# Bad return code from 'ipset list'. error=exit status 1 family="inet" stderr="ipset v7.1: Kernel and userspace incompatible: settype hash:ip,port with revision 6 not supported by userspace. +kube_proxy_mode: iptables + +# Test with SELinux in enforcing mode +preinstall_selinux_state: enforcing diff --git a/kubespray/tests/files/packet_fedora38-docker-calico.yml b/kubespray/tests/files/packet_fedora38-docker-calico.yml new file mode 100644 index 0000000..8d1a9a4 --- /dev/null +++ b/kubespray/tests/files/packet_fedora38-docker-calico.yml @@ -0,0 +1,15 @@ +--- +# Instance settings +cloud_image: fedora-38 +mode: default + +# Kubespray settings +auto_renew_certificates: true +# Switching to iptable due to https://github.com/projectcalico/calico/issues/5011 +# Kubernetes v1.23.0 kube-proxy does use v.7.x now. Calico v3.20.x/v3.21.x Pods show the following error +# Bad return code from 'ipset list'. error=exit status 1 family="inet" stderr="ipset v7.1: Kernel and userspace incompatible: settype hash:ip,port with revision 6 not supported by userspace. +kube_proxy_mode: iptables + +# Docker specific settings: +container_manager: docker +etcd_deployment_type: docker diff --git a/kubespray/tests/files/packet_fedora38-docker-weave.yml b/kubespray/tests/files/packet_fedora38-docker-weave.yml new file mode 100644 index 0000000..66bcdb1 --- /dev/null +++ b/kubespray/tests/files/packet_fedora38-docker-weave.yml @@ -0,0 +1,12 @@ +--- +# Instance settings +cloud_image: fedora-38 +mode: default + +# Kubespray settings +kube_network_plugin: weave + +# Docker specific settings: +container_manager: docker +etcd_deployment_type: docker +resolvconf_mode: docker_dns diff --git a/kubespray/tests/files/packet_fedora38-kube-ovn.yml b/kubespray/tests/files/packet_fedora38-kube-ovn.yml new file mode 100644 index 0000000..bc4ab5b --- /dev/null +++ b/kubespray/tests/files/packet_fedora38-kube-ovn.yml @@ -0,0 +1,7 @@ +--- +# Instance settings +cloud_image: fedora-38 +mode: default + +# Kubespray settings +kube_network_plugin: kube-ovn diff --git a/kubespray/tests/files/packet_opensuse-docker-cilium.yml b/kubespray/tests/files/packet_opensuse-docker-cilium.yml new file mode 100644 index 0000000..16ae393 --- /dev/null +++ b/kubespray/tests/files/packet_opensuse-docker-cilium.yml @@ -0,0 +1,11 @@ +--- +# Instance settings +cloud_image: opensuse-leap-15 +mode: default + +# Kubespray settings +kube_network_plugin: cilium + +# Docker specific settings: +container_manager: docker +etcd_deployment_type: docker diff --git a/kubespray/tests/files/packet_rockylinux8-calico.yml b/kubespray/tests/files/packet_rockylinux8-calico.yml new file mode 100644 index 0000000..b475112 --- /dev/null +++ b/kubespray/tests/files/packet_rockylinux8-calico.yml @@ -0,0 +1,11 @@ +--- +# Instance settings +cloud_image: rockylinux-8 +mode: default +vm_memory: 3072Mi + +# Kubespray settings +metrics_server_enabled: true +dashboard_namespace: "kube-dashboard" +dashboard_enabled: true +loadbalancer_apiserver_type: haproxy diff --git a/kubespray/tests/files/packet_rockylinux9-calico.yml b/kubespray/tests/files/packet_rockylinux9-calico.yml new file mode 100644 index 0000000..17e6ae5 --- /dev/null +++ b/kubespray/tests/files/packet_rockylinux9-calico.yml @@ -0,0 +1,11 @@ +--- +# Instance settings +cloud_image: rockylinux-9 +mode: default +vm_memory: 3072Mi + +# Kubespray settings +metrics_server_enabled: true +dashboard_namespace: "kube-dashboard" +dashboard_enabled: true +loadbalancer_apiserver_type: haproxy diff --git a/kubespray/tests/files/packet_rockylinux9-cilium.yml b/kubespray/tests/files/packet_rockylinux9-cilium.yml new file mode 100644 index 0000000..038e600 --- /dev/null +++ b/kubespray/tests/files/packet_rockylinux9-cilium.yml @@ -0,0 +1,10 @@ +--- +# Instance settings +cloud_image: rockylinux-9 +mode: default +vm_memory: 3072Mi + +# Kubespray settings +kube_network_plugin: cilium + +cilium_kube_proxy_replacement: strict diff --git a/kubespray/tests/files/packet_ubuntu20-aio-docker.yml b/kubespray/tests/files/packet_ubuntu20-aio-docker.yml new file mode 100644 index 0000000..edc1220 --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-aio-docker.yml @@ -0,0 +1,16 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: aio + +# Kubespray settings +auto_renew_certificates: true + +# Currently ipvs not available on KVM: https://packages.ubuntu.com/search?suite=focal&arch=amd64&mode=exactfilename&searchon=contents&keywords=ip_vs_sh.ko +kube_proxy_mode: iptables +enable_nodelocaldns: False + +# Use docker +container_manager: docker +etcd_deployment_type: docker +resolvconf_mode: docker_dns diff --git a/kubespray/tests/files/packet_ubuntu20-calico-aio-ansible-2_11.yml b/kubespray/tests/files/packet_ubuntu20-calico-aio-ansible-2_11.yml new file mode 120000 index 0000000..1006463 --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-calico-aio-ansible-2_11.yml @@ -0,0 +1 @@ +packet_ubuntu20-calico-aio.yml \ No newline at end of file diff --git a/kubespray/tests/files/packet_ubuntu20-calico-aio-hardening.yml b/kubespray/tests/files/packet_ubuntu20-calico-aio-hardening.yml new file mode 100644 index 0000000..16cf6ff --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-calico-aio-hardening.yml @@ -0,0 +1,106 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: aio + +# Kubespray settings +auto_renew_certificates: true + +# Currently ipvs not available on KVM: https://packages.ubuntu.com/search?suite=focal&arch=amd64&mode=exactfilename&searchon=contents&keywords=ip_vs_sh.ko +kube_proxy_mode: iptables +enable_nodelocaldns: False + +# The followings are for hardening +## kube-apiserver +authorization_modes: ['Node', 'RBAC'] +# AppArmor-based OS +kube_apiserver_feature_gates: ['AppArmor=true'] +kube_apiserver_request_timeout: 120s +kube_apiserver_service_account_lookup: true + +# enable kubernetes audit +kubernetes_audit: true +audit_log_path: "/var/log/kube-apiserver-log.json" +audit_log_maxage: 30 +audit_log_maxbackups: 10 +audit_log_maxsize: 100 + +tls_min_version: VersionTLS12 +tls_cipher_suites: + - TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 + - TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 + - TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305 + +# enable encryption at rest +kube_encrypt_secret_data: true +kube_encryption_resources: [secrets] +kube_encryption_algorithm: "secretbox" + +kube_apiserver_enable_admission_plugins: + - EventRateLimit + - AlwaysPullImages + - ServiceAccount + - NamespaceLifecycle + - NodeRestriction + - LimitRanger + - ResourceQuota + - MutatingAdmissionWebhook + - ValidatingAdmissionWebhook + - PodNodeSelector + - PodSecurity +kube_apiserver_admission_control_config_file: true +# EventRateLimit plugin configuration +kube_apiserver_admission_event_rate_limits: + limit_1: + type: Namespace + qps: 50 + burst: 100 + cache_size: 2000 + limit_2: + type: User + qps: 50 + burst: 100 +kube_profiling: false + +## kube-controller-manager +kube_controller_manager_bind_address: 127.0.0.1 +kube_controller_terminated_pod_gc_threshold: 50 +# AppArmor-based OS +kube_controller_feature_gates: ["RotateKubeletServerCertificate=true", "AppArmor=true"] + +## kube-scheduler +kube_scheduler_bind_address: 127.0.0.1 +# AppArmor-based OS +kube_scheduler_feature_gates: ["AppArmor=true"] + +## etcd +etcd_deployment_type: kubeadm + +## kubelet +kubelet_authentication_token_webhook: true +kube_read_only_port: 0 +kubelet_rotate_server_certificates: true +kubelet_csr_approver_enabled: false +kubelet_protect_kernel_defaults: true +kubelet_event_record_qps: 1 +kubelet_rotate_certificates: true +kubelet_streaming_connection_idle_timeout: "5m" +kubelet_make_iptables_util_chains: true +kubelet_feature_gates: ["RotateKubeletServerCertificate=true", "SeccompDefault=true"] +kubelet_seccomp_default: true +kubelet_systemd_hardening: true +# In case you have multiple interfaces in your +# control plane nodes and you want to specify the right +# IP addresses, kubelet_secure_addresses allows you +# to specify the IP from which the kubelet +# will receive the packets. +# kubelet_secure_addresses: "192.168.10.110 192.168.10.111 192.168.10.112" + +# additional configurations +kube_owner: root +kube_cert_group: root + +# create a default Pod Security Configuration and deny running of insecure pods +# kube-system namespace is exempted by default +kube_pod_security_use_default: true +kube_pod_security_default_enforce: restricted diff --git a/kubespray/tests/files/packet_ubuntu20-calico-aio.yml b/kubespray/tests/files/packet_ubuntu20-calico-aio.yml new file mode 100644 index 0000000..41d4a13 --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-calico-aio.yml @@ -0,0 +1,11 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: aio + +# Kubespray settings +auto_renew_certificates: true + +# Currently ipvs not available on KVM: https://packages.ubuntu.com/search?suite=focal&arch=amd64&mode=exactfilename&searchon=contents&keywords=ip_vs_sh.ko +kube_proxy_mode: iptables +enable_nodelocaldns: False diff --git a/kubespray/tests/files/packet_ubuntu20-calico-etcd-kubeadm-upgrade-ha.yml b/kubespray/tests/files/packet_ubuntu20-calico-etcd-kubeadm-upgrade-ha.yml new file mode 100644 index 0000000..57187a8 --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-calico-etcd-kubeadm-upgrade-ha.yml @@ -0,0 +1,24 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: ha + +# use the kubeadm etcd setting to test the upgrade +etcd_deployment_type: kubeadm + +upgrade_cluster_setup: true + +# Currently ipvs not available on KVM: https://packages.ubuntu.com/search?suite=focal&arch=amd64&mode=exactfilename&searchon=contents&keywords=ip_vs_sh.ko +kube_proxy_mode: iptables +enable_nodelocaldns: False + +# Pin disabling ipip mode to ensure proper upgrade +ipip: false +calico_vxlan_mode: Always +calico_network_backend: bird + +# Needed to bypass deprecation check +ignore_assert_errors: true +### FIXME FLORYUT Needed for upgrade job, will be removed when releasing kubespray 2.20 +calico_pool_blocksize: 24 +### /FIXME diff --git a/kubespray/tests/files/packet_ubuntu20-calico-etcd-kubeadm.yml b/kubespray/tests/files/packet_ubuntu20-calico-etcd-kubeadm.yml new file mode 100644 index 0000000..99f7365 --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-calico-etcd-kubeadm.yml @@ -0,0 +1,11 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: default + +# use the kubeadm etcd setting to test the upgrade +etcd_deployment_type: kubeadm + +# Currently ipvs not available on KVM: https://packages.ubuntu.com/search?suite=focal&arch=amd64&mode=exactfilename&searchon=contents&keywords=ip_vs_sh.ko +kube_proxy_mode: iptables +enable_nodelocaldns: False diff --git a/kubespray/tests/files/packet_ubuntu20-calico-ha-recover-noquorum.yml b/kubespray/tests/files/packet_ubuntu20-calico-ha-recover-noquorum.yml new file mode 100644 index 0000000..2d6db36 --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-calico-ha-recover-noquorum.yml @@ -0,0 +1,4 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: ha-recover-noquorum diff --git a/kubespray/tests/files/packet_ubuntu20-calico-ha-recover.yml b/kubespray/tests/files/packet_ubuntu20-calico-ha-recover.yml new file mode 100644 index 0000000..a757719 --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-calico-ha-recover.yml @@ -0,0 +1,4 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: ha-recover diff --git a/kubespray/tests/files/packet_ubuntu20-calico-ha-wireguard.yml b/kubespray/tests/files/packet_ubuntu20-calico-ha-wireguard.yml new file mode 100644 index 0000000..f2e2f57 --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-calico-ha-wireguard.yml @@ -0,0 +1,13 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: ha + +# Kubespray settings +calico_wireguard_enabled: true +auto_renew_certificates: true + +# Currently ipvs not available on KVM: https://packages.ubuntu.com/search?suite=focal&arch=amd64&mode=exactfilename&searchon=contents&keywords=ip_vs_sh.ko +kube_proxy_mode: iptables +# KVM kernel used by packet instances is missing the dummy.ko kernel module so it cannot enable nodelocaldns +enable_nodelocaldns: false diff --git a/kubespray/tests/files/packet_ubuntu20-cilium-sep.yml b/kubespray/tests/files/packet_ubuntu20-cilium-sep.yml new file mode 100644 index 0000000..104f7c3 --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-cilium-sep.yml @@ -0,0 +1,9 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: separate + +# Kubespray settings +kube_network_plugin: cilium +enable_network_policy: true +auto_renew_certificates: true diff --git a/kubespray/tests/files/packet_ubuntu20-crio.yml b/kubespray/tests/files/packet_ubuntu20-crio.yml new file mode 100644 index 0000000..87329d0 --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-crio.yml @@ -0,0 +1,10 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: default + +# Kubespray settings +container_manager: crio + +download_localhost: false +download_run_once: true diff --git a/kubespray/tests/files/packet_ubuntu20-docker-weave-sep.yml b/kubespray/tests/files/packet_ubuntu20-docker-weave-sep.yml new file mode 100644 index 0000000..8c6584c --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-docker-weave-sep.yml @@ -0,0 +1,16 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: separate + +# Kubespray settings +kube_network_plugin: weave +auto_renew_certificates: true + +# Docker specific settings: +container_manager: docker +etcd_deployment_type: docker +resolvconf_mode: docker_dns + +# Ubuntu 16 - docker containerd package available stopped at 1.4.6 +docker_containerd_version: latest diff --git a/kubespray/tests/files/packet_ubuntu20-flannel-ha-once.yml b/kubespray/tests/files/packet_ubuntu20-flannel-ha-once.yml new file mode 100644 index 0000000..4477421 --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-flannel-ha-once.yml @@ -0,0 +1,22 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: ha + +# Kubespray settings +kubeadm_certificate_key: 3998c58db6497dd17d909394e62d515368c06ec617710d02edea31c06d741085 +kube_proxy_mode: iptables +kube_network_plugin: flannel +helm_enabled: true +krew_enabled: true +kubernetes_audit: true +etcd_events_cluster_enabled: true +local_volume_provisioner_enabled: true +kube_encrypt_secret_data: true +ingress_nginx_enabled: true +cert_manager_enabled: true +# Disable as health checks are still unstable and slow to respond. +metrics_server_enabled: false +metrics_server_kubelet_insecure_tls: true +kube_token_auth: true +enable_nodelocaldns: false diff --git a/kubespray/tests/files/packet_ubuntu20-flannel-ha.yml b/kubespray/tests/files/packet_ubuntu20-flannel-ha.yml new file mode 100644 index 0000000..06a9ffb --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu20-flannel-ha.yml @@ -0,0 +1,10 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: ha + +# Kubespray settings +kube_network_plugin: flannel +etcd_deployment_type: kubeadm +kubeadm_certificate_key: 3998c58db6497dd17d909394e62d515368c06ec617710d02edea31c06d741085 +skip_non_kubeadm_warning: true diff --git a/kubespray/tests/files/packet_ubuntu22-aio-docker.yml b/kubespray/tests/files/packet_ubuntu22-aio-docker.yml new file mode 100644 index 0000000..b78c6b0 --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu22-aio-docker.yml @@ -0,0 +1,17 @@ +--- +# Instance settings +cloud_image: ubuntu-2204 +mode: aio +vm_memory: 1600Mi + +# Kubespray settings +auto_renew_certificates: true + +# Currently ipvs not available on KVM: https://packages.ubuntu.com/search?suite=focal&arch=amd64&mode=exactfilename&searchon=contents&keywords=ip_vs_sh.ko +kube_proxy_mode: iptables +enable_nodelocaldns: False + +# Use docker +container_manager: docker +etcd_deployment_type: docker +resolvconf_mode: docker_dns diff --git a/kubespray/tests/files/packet_ubuntu22-calico-aio.yml b/kubespray/tests/files/packet_ubuntu22-calico-aio.yml new file mode 100644 index 0000000..c9458f5 --- /dev/null +++ b/kubespray/tests/files/packet_ubuntu22-calico-aio.yml @@ -0,0 +1,27 @@ +--- +# Instance settings +cloud_image: ubuntu-2204 +mode: aio +vm_memory: 1600Mi + +# Kubespray settings +auto_renew_certificates: true + +# Currently ipvs not available on KVM: https://packages.ubuntu.com/search?suite=focal&arch=amd64&mode=exactfilename&searchon=contents&keywords=ip_vs_sh.ko +kube_proxy_mode: iptables +enable_nodelocaldns: False + +containerd_registries: + "docker.io": "https://mirror.gcr.io" + +containerd_registries_mirrors: + - prefix: docker.io + mirrors: + - host: https://mirror.gcr.io + capabilities: ["pull", "resolve"] + skip_verify: false + - prefix: 172.19.16.11:5000 + mirrors: + - host: http://172.19.16.11:5000 + capabilities: ["pull", "resolve", "push"] + skip_verify: true diff --git a/kubespray/tests/files/tf-elastx_ubuntu20-calico.yml b/kubespray/tests/files/tf-elastx_ubuntu20-calico.yml new file mode 100644 index 0000000..b8dbaaa --- /dev/null +++ b/kubespray/tests/files/tf-elastx_ubuntu20-calico.yml @@ -0,0 +1,5 @@ +--- +sonobuoy_enabled: true + +# Ignore ping errors +ignore_assert_errors: true diff --git a/kubespray/tests/files/tf-ovh_ubuntu20-calico.yml b/kubespray/tests/files/tf-ovh_ubuntu20-calico.yml new file mode 100644 index 0000000..d6fb9de --- /dev/null +++ b/kubespray/tests/files/tf-ovh_ubuntu20-calico.yml @@ -0,0 +1,7 @@ +--- +sonobuoy_enabled: true +pkg_install_retries: 25 +retry_stagger: 10 + +# Ignore ping errors +ignore_assert_errors: true diff --git a/kubespray/tests/files/vagrant_centos7-kube-router.rb b/kubespray/tests/files/vagrant_centos7-kube-router.rb new file mode 100644 index 0000000..620df71 --- /dev/null +++ b/kubespray/tests/files/vagrant_centos7-kube-router.rb @@ -0,0 +1,15 @@ +$num_instances = 2 +$vm_memory ||= 2048 +$os = "centos" + +$kube_master_instances = 1 +$etcd_instances = 1 + +# For CI we are not worried about data persistence across reboot +$libvirt_volume_cache = "unsafe" + +# Checking for box update can trigger API rate limiting +# https://www.vagrantup.com/docs/vagrant-cloud/request-limits.html +$box_check_update = false + +$network_plugin = "kube-router" diff --git a/kubespray/tests/files/vagrant_centos7-kube-router.yml b/kubespray/tests/files/vagrant_centos7-kube-router.yml new file mode 100644 index 0000000..e9e4161 --- /dev/null +++ b/kubespray/tests/files/vagrant_centos7-kube-router.yml @@ -0,0 +1,8 @@ +--- +# Instance settings +cloud_image: centos-7 +mode: default + +# Kubespray settings +kube_network_plugin: kube-router +enable_network_policy: true diff --git a/kubespray/tests/files/vagrant_fedora37-kube-router.rb b/kubespray/tests/files/vagrant_fedora37-kube-router.rb new file mode 100644 index 0000000..54bcd75 --- /dev/null +++ b/kubespray/tests/files/vagrant_fedora37-kube-router.rb @@ -0,0 +1,15 @@ +$num_instances = 2 +$vm_memory ||= 2048 +$os = "fedora37" + +$kube_master_instances = 1 +$etcd_instances = 1 + +# For CI we are not worried about data persistence across reboot +$libvirt_volume_cache = "unsafe" + +# Checking for box update can trigger API rate limiting +# https://www.vagrantup.com/docs/vagrant-cloud/request-limits.html +$box_check_update = false + +$network_plugin = "kube-router" diff --git a/kubespray/tests/files/vagrant_fedora37-kube-router.yml b/kubespray/tests/files/vagrant_fedora37-kube-router.yml new file mode 100644 index 0000000..2cee421 --- /dev/null +++ b/kubespray/tests/files/vagrant_fedora37-kube-router.yml @@ -0,0 +1,7 @@ +--- +# Instance settings +cloud_image: fedora-37 +mode: default + +# Kubespray settings +kube_network_plugin: kube-router diff --git a/kubespray/tests/files/vagrant_ubuntu20-calico-dual-stack.rb b/kubespray/tests/files/vagrant_ubuntu20-calico-dual-stack.rb new file mode 100644 index 0000000..f7d7765 --- /dev/null +++ b/kubespray/tests/files/vagrant_ubuntu20-calico-dual-stack.rb @@ -0,0 +1,7 @@ +# For CI we are not worried about data persistence across reboot +$libvirt_volume_cache = "unsafe" + +# Checking for box update can trigger API rate limiting +# https://www.vagrantup.com/docs/vagrant-cloud/request-limits.html +$box_check_update = false +$network_plugin = "calico" diff --git a/kubespray/tests/files/vagrant_ubuntu20-calico-dual-stack.yml b/kubespray/tests/files/vagrant_ubuntu20-calico-dual-stack.yml new file mode 100644 index 0000000..3a45bdc --- /dev/null +++ b/kubespray/tests/files/vagrant_ubuntu20-calico-dual-stack.yml @@ -0,0 +1,3 @@ +--- +# Kubespray settings +enable_dual_stack_networks: true diff --git a/kubespray/tests/files/vagrant_ubuntu20-flannel-collection.rb b/kubespray/tests/files/vagrant_ubuntu20-flannel-collection.rb new file mode 100644 index 0000000..c739f58 --- /dev/null +++ b/kubespray/tests/files/vagrant_ubuntu20-flannel-collection.rb @@ -0,0 +1,9 @@ +$os = "ubuntu2004" + +# For CI we are not worries about data persistence across reboot +$libvirt_volume_cache = "unsafe" + +# Checking for box update can trigger API rate limiting +# https://www.vagrantup.com/docs/vagrant-cloud/request-limits.html +$box_check_update = false +$vm_cpus = 2 \ No newline at end of file diff --git a/kubespray/tests/files/vagrant_ubuntu20-flannel-collection.yml b/kubespray/tests/files/vagrant_ubuntu20-flannel-collection.yml new file mode 100644 index 0000000..6f8916f --- /dev/null +++ b/kubespray/tests/files/vagrant_ubuntu20-flannel-collection.yml @@ -0,0 +1,3 @@ +--- +# Kubespray settings +kube_network_plugin: flannel diff --git a/kubespray/tests/files/vagrant_ubuntu20-flannel.rb b/kubespray/tests/files/vagrant_ubuntu20-flannel.rb new file mode 100644 index 0000000..55daa19 --- /dev/null +++ b/kubespray/tests/files/vagrant_ubuntu20-flannel.rb @@ -0,0 +1,9 @@ +$os = "ubuntu2004" + +# For CI we are not worries about data persistence across reboot +$libvirt_volume_cache = "unsafe" + +# Checking for box update can trigger API rate limiting +# https://www.vagrantup.com/docs/vagrant-cloud/request-limits.html +$box_check_update = false +$vm_cpus = 2 diff --git a/kubespray/tests/files/vagrant_ubuntu20-flannel.yml b/kubespray/tests/files/vagrant_ubuntu20-flannel.yml new file mode 100644 index 0000000..6f8916f --- /dev/null +++ b/kubespray/tests/files/vagrant_ubuntu20-flannel.yml @@ -0,0 +1,3 @@ +--- +# Kubespray settings +kube_network_plugin: flannel diff --git a/kubespray/tests/files/vagrant_ubuntu20-kube-router-sep.rb b/kubespray/tests/files/vagrant_ubuntu20-kube-router-sep.rb new file mode 100644 index 0000000..999f813 --- /dev/null +++ b/kubespray/tests/files/vagrant_ubuntu20-kube-router-sep.rb @@ -0,0 +1,15 @@ +$num_instances = 2 +$vm_memory ||= 2048 +$os = "ubuntu2004" + +$kube_master_instances = 1 +$etcd_instances = 1 + +# For CI we are not worried about data persistence across reboot +$libvirt_volume_cache = "unsafe" + +# Checking for box update can trigger API rate limiting +# https://www.vagrantup.com/docs/vagrant-cloud/request-limits.html +$box_check_update = false + +$network_plugin = "kube-router" diff --git a/kubespray/tests/files/vagrant_ubuntu20-kube-router-sep.yml b/kubespray/tests/files/vagrant_ubuntu20-kube-router-sep.yml new file mode 100644 index 0000000..d17b627 --- /dev/null +++ b/kubespray/tests/files/vagrant_ubuntu20-kube-router-sep.yml @@ -0,0 +1,8 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: separate + +# Kubespray settings +bootstrap_os: ubuntu +kube_network_plugin: kube-router diff --git a/kubespray/tests/files/vagrant_ubuntu20-kube-router-svc-proxy.rb b/kubespray/tests/files/vagrant_ubuntu20-kube-router-svc-proxy.rb new file mode 100644 index 0000000..29f6e81 --- /dev/null +++ b/kubespray/tests/files/vagrant_ubuntu20-kube-router-svc-proxy.rb @@ -0,0 +1,10 @@ +$os = "ubuntu2004" + +# For CI we are not worried about data persistence across reboot +$libvirt_volume_cache = "unsafe" + +# Checking for box update can trigger API rate limiting +# https://www.vagrantup.com/docs/vagrant-cloud/request-limits.html +$box_check_update = false + +$network_plugin = "kube-router" diff --git a/kubespray/tests/files/vagrant_ubuntu20-kube-router-svc-proxy.yml b/kubespray/tests/files/vagrant_ubuntu20-kube-router-svc-proxy.yml new file mode 100644 index 0000000..faa30d0 --- /dev/null +++ b/kubespray/tests/files/vagrant_ubuntu20-kube-router-svc-proxy.yml @@ -0,0 +1,10 @@ +--- +# Instance settings +cloud_image: ubuntu-2004 +mode: separate + +# Kubespray settings +bootstrap_os: ubuntu +kube_network_plugin: kube-router + +kube_router_run_service_proxy: true diff --git a/kubespray/tests/files/vagrant_ubuntu20-weave-medium.rb b/kubespray/tests/files/vagrant_ubuntu20-weave-medium.rb new file mode 100644 index 0000000..6cf49c9 --- /dev/null +++ b/kubespray/tests/files/vagrant_ubuntu20-weave-medium.rb @@ -0,0 +1,7 @@ +$num_instances = 16 +$vm_memory ||= 2048 +$os = "ubuntu2004" +$network_plugin = "weave" +$kube_master_instances = 1 +$etcd_instances = 1 +$playbook = "tests/cloud_playbooks/wait-for-ssh.yml" diff --git a/kubespray/tests/files/vagrant_ubuntu20-weave-medium.yml b/kubespray/tests/files/vagrant_ubuntu20-weave-medium.yml new file mode 100644 index 0000000..bb5f974 --- /dev/null +++ b/kubespray/tests/files/vagrant_ubuntu20-weave-medium.yml @@ -0,0 +1,3 @@ +--- +# Kubespray settings +kube_network_plugin: weave diff --git a/kubespray/tests/local_inventory/host_vars/localhost b/kubespray/tests/local_inventory/host_vars/localhost new file mode 100644 index 0000000..695c0ec --- /dev/null +++ b/kubespray/tests/local_inventory/host_vars/localhost @@ -0,0 +1,12 @@ +aws: + key_name: "{{ key_name | default('ansibl8s') }}" + access_key: "{{ aws_access_key }}" + secret_key: "{{ aws_secret_key }}" + region: "{{ aws_region | default('eu-west-1') }}" # default to eu-west-1 + group: "{{ aws_security_group | default ('default')}}" + instance_type: t2.micro + ami_id: "{{ aws_ami_id | default('ami-02724d1f') }}" # default to Debian Jessie + count: 3 + tags: + test_id: "{{ test_id }}" + network_plugin: "{{ kube_network_plugin }}" diff --git a/kubespray/tests/local_inventory/hosts.cfg b/kubespray/tests/local_inventory/hosts.cfg new file mode 100644 index 0000000..2302eda --- /dev/null +++ b/kubespray/tests/local_inventory/hosts.cfg @@ -0,0 +1 @@ +localhost ansible_connection=local diff --git a/kubespray/tests/requirements.txt b/kubespray/tests/requirements.txt new file mode 100644 index 0000000..19474ab --- /dev/null +++ b/kubespray/tests/requirements.txt @@ -0,0 +1,11 @@ +-r ../requirements.txt +ansible-lint==6.16.2 +apache-libcloud==3.7.0 +ara[server]==1.6.1 +dopy==0.3.7 +molecule==5.0.1 +molecule-plugins[vagrant]==23.4.1 +python-vagrant==1.0.0 +pytest-testinfra==8.1.0 +tox==4.5.2 +yamllint==1.32.0 diff --git a/kubespray/tests/requirements.yml b/kubespray/tests/requirements.yml new file mode 100644 index 0000000..2bedd23 --- /dev/null +++ b/kubespray/tests/requirements.yml @@ -0,0 +1,4 @@ +--- +collections: + - name: amazon.aws + version: 6.0.1 diff --git a/kubespray/tests/run-tests.sh b/kubespray/tests/run-tests.sh new file mode 100755 index 0000000..c20438e --- /dev/null +++ b/kubespray/tests/run-tests.sh @@ -0,0 +1,8 @@ +#! /bin/bash + +# curl -# -C - -o shebang-unit https://raw.github.com/arpinum-oss/shebang-unit/master/releases/shebang-unit +# chmod +x shebang-unit + +now=$(date +"%Y%m%d%H%M%S") +mkdir -p ${PWD}/tests-results +./shebang-unit --reporters=simple,junit --output-file=${PWD}/tests-results/junit_report-${now}.xml tests diff --git a/kubespray/tests/scripts/ansibl8s_test.sh b/kubespray/tests/scripts/ansibl8s_test.sh new file mode 100644 index 0000000..1f61f45 --- /dev/null +++ b/kubespray/tests/scripts/ansibl8s_test.sh @@ -0,0 +1,52 @@ +#! /bin/bash + +global_setup() { + git clone https://github.com/ansibl8s/setup-kubernetes.git setup-kubernetes + private_key="" + if [ ! -z ${PRIVATE_KEY_FILE} ] + then + private_key="--private-key=${PRIVATE_KEY_FILE}" + fi + ansible-playbook create.yml -i hosts -u admin -s \ + -e test_id=${TEST_ID} \ + -e kube_network_plugin=${KUBE_NETWORK_PLUGIN} \ + -e aws_access_key=${AWS_ACCESS_KEY} \ + -e aws_secret_key=${AWS_SECRET_KEY} \ + -e aws_ami_id=${AWS_AMI_ID} \ + -e aws_security_group=${AWS_SECURITY_GROUP} \ + -e key_name=${AWS_KEY_PAIR_NAME} \ + -e inventory_path=${PWD}/inventory.ini \ + -e aws_region=${AWS_REGION} +} + +global_teardown() { + if [ -f inventory.ini ]; + then + ansible-playbook -i inventory.ini -u admin delete.yml + fi + rm -rf ${PWD}/setup-kubernetes +} + +should_deploy_cluster() { + ansible-playbook -i inventory.ini -s ${private_key} -e kube_network_plugin=${KUBE_NETWORK_PLUGIN} setup-kubernetes/cluster.yml + + assertion__status_code_is_success $? +} + +should_api_server_respond() { + ansible-playbook -i inventory.ini ${private_key} testcases/010_check-apiserver.yml + + assertion__status_code_is_success $? +} + +should_pod_be_in_expected_subnet() { + ansible-playbook -i inventory.ini -s ${private_key} testcases/030_check-network.yml -vv + + assertion__status_code_is_success $? +} + +should_resolve_cluster_dns() { + ansible-playbook -i inventory.ini -s ${private_key} testcases/040_check-network-adv.yml -vv + + assertion__status_code_is_success $? +} diff --git a/kubespray/tests/scripts/check_galaxy_version.sh b/kubespray/tests/scripts/check_galaxy_version.sh new file mode 100755 index 0000000..b6679db --- /dev/null +++ b/kubespray/tests/scripts/check_galaxy_version.sh @@ -0,0 +1,18 @@ +#!/bin/bash +set -e + +version_from_galaxy=$(grep "^version:" galaxy.yml | awk '{print $2}') +version_from_docs=$(grep -P "^\s+version:\sv\d+\.\d+\.\d+" docs/ansible_collection.md | awk '{print $2}') + +if [[ $KUBESPRAY_VERSION != "v${version_from_galaxy}" ]] +then + echo "Please update galaxy.yml version to match the KUBESPRAY_VERSION. Be sure to remove the \"v\" to adhere" + echo "to semenatic versioning" + exit 1 +fi + +if [[ $KUBESPRAY_VERSION != "${version_from_docs}" ]] +then + echo "Please update the documentation for Ansible collections under docs/ansible_collection.md to reflect the KUBESPRAY_VERSION" + exit 1 +fi diff --git a/kubespray/tests/scripts/check_readme_versions.sh b/kubespray/tests/scripts/check_readme_versions.sh new file mode 100755 index 0000000..d796d9c --- /dev/null +++ b/kubespray/tests/scripts/check_readme_versions.sh @@ -0,0 +1,33 @@ +#!/bin/bash +set -e + +TARGET_COMPONENTS="containerd calico cilium flannel kube-ovn kube-router weave cert-manager krew helm metallb registry cephfs-provisioner rbd-provisioner aws-ebs-csi-plugin azure-csi-plugin cinder-csi-plugin gcp-pd-csi-plugin local-path-provisioner local-volume-provisioner kube-vip ingress-nginx" + +# cd to the root directory of kubespray +cd $(dirname $0)/../../ + +echo checking kubernetes.. +version_from_default=$(grep "^kube_version:" ./roles/kubespray-defaults/defaults/main.yaml | awk '{print $2}' | sed s/\"//g) +version_from_readme=$(grep " \[kubernetes\]" ./README.md | awk '{print $3}') +if [ "${version_from_default}" != "${version_from_readme}" ]; then + echo "The version of kubernetes is different between main.yml(${version_from_default}) and README.md(${version_from_readme})." + echo "If the pull request updates kubernetes version, please update README.md also." + exit 1 +fi + +for component in $(echo ${TARGET_COMPONENTS}); do + echo checking ${component}.. + version_from_default=$(grep "^$(echo ${component} | sed s/"-"/"_"/g)_version:" ./roles/download/defaults/main/main.yml | awk '{print $2}' | sed s/\"//g | sed s/^v//) + if [ "${version_from_default}" = "" ]; then + version_from_default=$(grep "^$(echo ${component} | sed s/"-"/"_"/g)_version:" ./roles/kubernetes/node/defaults/main.yml | awk '{print $2}' | sed s/\"//g | sed s/^v//) + fi + version_from_readme=$(grep "\[${component}\]" ./README.md | grep "https" | awk '{print $3}' | sed s/^v//) + if [ "${version_from_default}" != "${version_from_readme}" ]; then + echo "The version of ${component} is different between main.yml(${version_from_default}) and README.md(${version_from_readme})." + echo "If the pull request updates ${component} version, please update README.md also." + exit 1 + fi +done + +echo "Succeeded to check all components." +exit 0 diff --git a/kubespray/tests/scripts/check_typo.sh b/kubespray/tests/scripts/check_typo.sh new file mode 100755 index 0000000..522d4b2 --- /dev/null +++ b/kubespray/tests/scripts/check_typo.sh @@ -0,0 +1,12 @@ +#!/bin/bash + +# cd to the root directory of kubespray +cd $(dirname $0)/../../ + +rm ./misspell* + +set -e +wget https://github.com/client9/misspell/releases/download/v0.3.4/misspell_0.3.4_linux_64bit.tar.gz +tar -zxvf ./misspell_0.3.4_linux_64bit.tar.gz +chmod 755 ./misspell +git ls-files | grep -v OWNERS_ALIASES | xargs ./misspell -error diff --git a/kubespray/tests/scripts/create-tf.sh b/kubespray/tests/scripts/create-tf.sh new file mode 100755 index 0000000..fbed302 --- /dev/null +++ b/kubespray/tests/scripts/create-tf.sh @@ -0,0 +1,5 @@ +#!/bin/bash +set -euxo pipefail + +cd .. +terraform -chdir="contrib/terraform/$PROVIDER" apply -auto-approve -parallelism=1 diff --git a/kubespray/tests/scripts/delete-tf.sh b/kubespray/tests/scripts/delete-tf.sh new file mode 100755 index 0000000..57c35c8 --- /dev/null +++ b/kubespray/tests/scripts/delete-tf.sh @@ -0,0 +1,5 @@ +#!/bin/bash +set -euxo pipefail + +cd .. +terraform -chdir="contrib/terraform/$PROVIDER" destroy -auto-approve diff --git a/kubespray/tests/scripts/md-table/main.py b/kubespray/tests/scripts/md-table/main.py new file mode 100755 index 0000000..9e00005 --- /dev/null +++ b/kubespray/tests/scripts/md-table/main.py @@ -0,0 +1,96 @@ +#!/usr/bin/env python +import argparse +import sys +import glob +from pathlib import Path +import yaml +from pydblite import Base +import re +import jinja2 +import sys + +from pprint import pprint + + +parser = argparse.ArgumentParser(description='Generate a Markdown table representing the CI test coverage') +parser.add_argument('--dir', default='tests/files/', help='folder with test yml files') + + +args = parser.parse_args() +p = Path(args.dir) + +env = jinja2.Environment(loader=jinja2.FileSystemLoader(searchpath=sys.path[0])) + +# Data represents CI coverage data matrix +class Data: + def __init__(self): + self.db = Base(':memory:') + self.db.create('container_manager', 'network_plugin', 'operating_system') + + + def set(self, container_manager, network_plugin, operating_system): + self.db.insert(container_manager=container_manager, network_plugin=network_plugin, operating_system=operating_system) + self.db.commit() + def exists(self, container_manager, network_plugin, operating_system): + return len((self.db("container_manager") == container_manager) & (self.db("network_plugin") == network_plugin) & (self.db("operating_system") == operating_system)) > 0 + + def jinja(self): + template = env.get_template('table.md.j2') + container_engines = list(self.db.get_unique_ids('container_manager')) + network_plugins = list(self.db.get_unique_ids("network_plugin")) + operating_systems = list(self.db.get_unique_ids("operating_system")) + + container_engines.sort() + network_plugins.sort() + operating_systems.sort() + + return template.render( + container_engines=container_engines, + network_plugins=network_plugins, + operating_systems=operating_systems, + exists=self.exists + ) + + def markdown(self): + out = '' + for container_manager in self.db.get_unique_ids('container_manager'): + # Prepare the headers + out += "# " + container_manager + "\n" + headers = '|OS / CNI| ' + underline = '|----|' + for network_plugin in self.db.get_unique_ids("network_plugin"): + headers += network_plugin + ' | ' + underline += '----|' + out += headers + "\n" + underline + "\n" + for operating_system in self.db.get_unique_ids("operating_system"): + out += '| ' + operating_system + ' | ' + for network_plugin in self.db.get_unique_ids("network_plugin"): + if self.exists(container_manager, network_plugin, operating_system): + emoji = ':white_check_mark:' + else: + emoji = ':x:' + out += emoji + ' | ' + out += "\n" + + pprint(self.db.get_unique_ids('operating_system')) + pprint(self.db.get_unique_ids('network_plugin')) + return out + + + +if not p.is_dir(): + print("Path is not a directory") + sys.exit(2) + +data = Data() +files = p.glob('*.yml') +for f in files: + y = yaml.load(f.open(), Loader=yaml.FullLoader) + + container_manager = y.get('container_manager', 'containerd') + network_plugin = y.get('kube_network_plugin', 'calico') + x = re.match(r"^[a-z-]+_([a-z0-9]+).*", f.name) + operating_system = x.group(1) + data.set(container_manager=container_manager, network_plugin=network_plugin, operating_system=operating_system) +#print(data.markdown()) +print(data.jinja()) diff --git a/kubespray/tests/scripts/md-table/requirements.txt b/kubespray/tests/scripts/md-table/requirements.txt new file mode 100644 index 0000000..6d4aca3 --- /dev/null +++ b/kubespray/tests/scripts/md-table/requirements.txt @@ -0,0 +1,4 @@ +jinja2 +pathlib ; python_version < '3.10' +pyaml +pydblite diff --git a/kubespray/tests/scripts/md-table/table.md.j2 b/kubespray/tests/scripts/md-table/table.md.j2 new file mode 100644 index 0000000..c5cf37d --- /dev/null +++ b/kubespray/tests/scripts/md-table/table.md.j2 @@ -0,0 +1,15 @@ +# CI test coverage + +To generate this Matrix run `./tests/scripts/md-table/main.py` + +{%- for container_engine in container_engines %} + +## {{ container_engine }} + +| OS / CNI |{% for cni in network_plugins %} {{ cni }} |{% endfor %} +|---|{% for cni in network_plugins %} --- |{% endfor %} +{%- for os in operating_systems %} +{{ os }} | {% for cni in network_plugins %} {{ ':white_check_mark:' if exists(container_engine, cni, os) else ':x:' }} |{% endfor %} +{%- endfor %} + +{%- endfor %} diff --git a/kubespray/tests/scripts/md-table/test.sh b/kubespray/tests/scripts/md-table/test.sh new file mode 100755 index 0000000..46daa63 --- /dev/null +++ b/kubespray/tests/scripts/md-table/test.sh @@ -0,0 +1,11 @@ +#!/bin/bash +set -euxo pipefail + +echo "Install requirements..." +pip install -r ./tests/scripts/md-table/requirements.txt + +echo "Generate current file..." +./tests/scripts/md-table/main.py > tmp.md + +echo "Compare docs/ci.md with actual tests in tests/files/*.yml ..." +cmp docs/ci.md tmp.md diff --git a/kubespray/tests/scripts/molecule_logs.sh b/kubespray/tests/scripts/molecule_logs.sh new file mode 100755 index 0000000..4908d81 --- /dev/null +++ b/kubespray/tests/scripts/molecule_logs.sh @@ -0,0 +1,9 @@ +#!/bin/bash + +# Ensure a clean environent +rm -fr molecule_logs +mkdir -p molecule_logs + +# Collect and archive the logs +find ~/.cache/molecule/ -name \*.out -o -name \*.err -type f | xargs tar -uf molecule_logs/molecule.tar +gzip molecule_logs/molecule.tar diff --git a/kubespray/tests/scripts/molecule_run.sh b/kubespray/tests/scripts/molecule_run.sh new file mode 100755 index 0000000..9604238 --- /dev/null +++ b/kubespray/tests/scripts/molecule_run.sh @@ -0,0 +1,34 @@ +#!/bin/bash +set -euxo pipefail -o noglob + +export LC_ALL=C.UTF-8 +export LANG=C.UTF-8 + +_PATH='roles' +_EXCLUDE="" + +while [[ $# -gt 0 ]] ; do + case $1 in + -e|--exclude) + _EXCLUDE="${_EXCLUDE} -not -path ${_PATH}/$2/*" + shift + shift + ;; + -i|--include) + _PATH="${_PATH}/$2" + shift + shift + ;; + -h|--help) + echo "Usage: molecule_run.sh [-h|--help] [-e|--exclude] [-i|--include]" + exit 0 + ;; + esac +done + +for d in $(find ${_PATH} ${_EXCLUDE} -name molecule -type d) +do + pushd $(dirname $d) + molecule test --all + popd +done diff --git a/kubespray/tests/scripts/rebase.sh b/kubespray/tests/scripts/rebase.sh new file mode 100755 index 0000000..36cb7f6 --- /dev/null +++ b/kubespray/tests/scripts/rebase.sh @@ -0,0 +1,15 @@ +#!/bin/bash +set -euxo pipefail + +KUBESPRAY_NEXT_VERSION=2.$(( ${KUBESPRAY_VERSION:3:2} + 1 )) + +# Rebase PRs on master (or release branch) to get latest changes +if [[ $CI_COMMIT_REF_NAME == pr-* ]]; then + git config user.email "ci@kubespray.io" + git config user.name "CI" + if [[ -z "`git branch -a --list origin/release-$KUBESPRAY_NEXT_VERSION`" ]]; then + git pull --rebase origin master + else + git pull --rebase origin release-$KUBESPRAY_NEXT_VERSION + fi +fi diff --git a/kubespray/tests/scripts/terraform_install.sh b/kubespray/tests/scripts/terraform_install.sh new file mode 100755 index 0000000..4228bbd --- /dev/null +++ b/kubespray/tests/scripts/terraform_install.sh @@ -0,0 +1,6 @@ +#!/bin/bash +set -euxo pipefail + +apt-get install -y unzip +curl https://releases.hashicorp.com/terraform/${TF_VERSION}/terraform_${TF_VERSION}_linux_amd64.zip > /tmp/terraform.zip +unzip /tmp/terraform.zip && mv ./terraform /usr/local/bin/ && terraform --version diff --git a/kubespray/tests/scripts/testcases_cleanup.sh b/kubespray/tests/scripts/testcases_cleanup.sh new file mode 100755 index 0000000..71b7fdc --- /dev/null +++ b/kubespray/tests/scripts/testcases_cleanup.sh @@ -0,0 +1,9 @@ +#!/bin/bash +set -euxo pipefail + +cd tests && make delete-${CI_PLATFORM} -s ; cd - + +if [ -d ~/.ara ] ; then + tar czvf ${CI_PROJECT_DIR}/cluster-dump/ara.tgz ~/.ara + rm -fr ~/.ara +fi diff --git a/kubespray/tests/scripts/testcases_prepare.sh b/kubespray/tests/scripts/testcases_prepare.sh new file mode 100755 index 0000000..aa4b285 --- /dev/null +++ b/kubespray/tests/scripts/testcases_prepare.sh @@ -0,0 +1,7 @@ +#!/bin/bash +set -euxo pipefail + +mkdir -p /.ssh +mkdir -p cluster-dump +mkdir -p $HOME/.ssh +ansible-playbook --version diff --git a/kubespray/tests/scripts/testcases_run.sh b/kubespray/tests/scripts/testcases_run.sh new file mode 100755 index 0000000..7cd4671 --- /dev/null +++ b/kubespray/tests/scripts/testcases_run.sh @@ -0,0 +1,174 @@ +#!/bin/bash +set -euxo pipefail + +echo "CI_JOB_NAME is $CI_JOB_NAME" +CI_TEST_ADDITIONAL_VARS="" + +if [[ "$CI_JOB_NAME" =~ "upgrade" ]]; then + if [ "${UPGRADE_TEST}" == "false" ]; then + echo "Job name contains 'upgrade', but UPGRADE_TEST='false'" + exit 1 + fi +else + if [ "${UPGRADE_TEST}" != "false" ]; then + echo "UPGRADE_TEST!='false', but job names does not contain 'upgrade'" + exit 1 + fi +fi + +# needed for ara not to complain +export TZ=UTC + +export ANSIBLE_REMOTE_USER=$SSH_USER +export ANSIBLE_BECOME=true +export ANSIBLE_BECOME_USER=root +export ANSIBLE_CALLBACK_PLUGINS="$(python -m ara.setup.callback_plugins)" + +cd tests && make create-${CI_PLATFORM} -s ; cd - +ansible-playbook tests/cloud_playbooks/wait-for-ssh.yml + +# Flatcar Container Linux needs auto update disabled +if [[ "$CI_JOB_NAME" =~ "coreos" ]]; then + ansible all -m raw -a 'systemctl disable locksmithd' + ansible all -m raw -a 'systemctl stop locksmithd' + mkdir -p /opt/bin && ln -s /usr/bin/python /opt/bin/python +fi + +if [[ "$CI_JOB_NAME" =~ "opensuse" ]]; then + # OpenSUSE needs netconfig update to get correct resolv.conf + # See https://goinggnu.wordpress.com/2013/10/14/how-to-fix-the-dns-in-opensuse-13-1/ + ansible all -m raw -a 'netconfig update -f' + # Auto import repo keys + ansible all -m raw -a 'zypper --gpg-auto-import-keys refresh' +fi + +if [[ "$CI_JOB_NAME" =~ "ubuntu" ]]; then + # We need to tell ansible that ubuntu hosts are python3 only + CI_TEST_ADDITIONAL_VARS="-e ansible_python_interpreter=/usr/bin/python3" +fi + +ENABLE_020_TEST="true" +ENABLE_030_TEST="true" +ENABLE_040_TEST="true" +if [[ "$CI_JOB_NAME" =~ "macvlan" ]]; then + ENABLE_020_TEST="false" + ENABLE_030_TEST="false" + ENABLE_040_TEST="false" +fi + +if [[ "$CI_JOB_NAME" =~ "hardening" ]]; then + # TODO: We need to remove this condition by finding alternative container + # image instead of netchecker which doesn't work at hardening environments. + ENABLE_040_TEST="false" +fi + +# Check out latest tag if testing upgrade +test "${UPGRADE_TEST}" != "false" && git fetch --all && git checkout "$KUBESPRAY_VERSION" +# Checkout the CI vars file so it is available +test "${UPGRADE_TEST}" != "false" && git checkout "${CI_COMMIT_SHA}" tests/files/${CI_JOB_NAME}.yml +test "${UPGRADE_TEST}" != "false" && git checkout "${CI_COMMIT_SHA}" ${CI_TEST_REGISTRY_MIRROR} +test "${UPGRADE_TEST}" != "false" && git checkout "${CI_COMMIT_SHA}" ${CI_TEST_SETTING} + +# Create cluster +ansible-playbook ${ANSIBLE_LOG_LEVEL} -e @${CI_TEST_SETTING} -e @${CI_TEST_REGISTRY_MIRROR} -e @${CI_TEST_VARS} -e local_release_dir=${PWD}/downloads --limit "all:!fake_hosts" cluster.yml + +# Repeat deployment if testing upgrade +if [ "${UPGRADE_TEST}" != "false" ]; then + test "${UPGRADE_TEST}" == "basic" && PLAYBOOK="cluster.yml" + test "${UPGRADE_TEST}" == "graceful" && PLAYBOOK="upgrade-cluster.yml" + git checkout "${CI_COMMIT_SHA}" + ansible-playbook ${ANSIBLE_LOG_LEVEL} -e @${CI_TEST_SETTING} -e @${CI_TEST_REGISTRY_MIRROR} -e @${CI_TEST_VARS} -e local_release_dir=${PWD}/downloads --limit "all:!fake_hosts" $PLAYBOOK +fi + +# Test control plane recovery +if [ "${RECOVER_CONTROL_PLANE_TEST}" != "false" ]; then + ansible-playbook ${ANSIBLE_LOG_LEVEL} -e @${CI_TEST_SETTING} -e @${CI_TEST_REGISTRY_MIRROR} -e @${CI_TEST_VARS} -e local_release_dir=${PWD}/downloads --limit "${RECOVER_CONTROL_PLANE_TEST_GROUPS}:!fake_hosts" -e reset_confirmation=yes reset.yml + ansible-playbook ${ANSIBLE_LOG_LEVEL} -e @${CI_TEST_SETTING} -e @${CI_TEST_REGISTRY_MIRROR} -e @${CI_TEST_VARS} -e local_release_dir=${PWD}/downloads -e etcd_retries=10 --limit "etcd:kube_control_plane:!fake_hosts" recover-control-plane.yml +fi + +# Test collection build and install by installing our collection, emptying our repository, adding +# cluster.yml, reset.yml, and remote-node.yml files that simply point to our collection's playbooks, and then +# running the same tests as before +if [[ "${CI_JOB_NAME}" =~ "collection" ]]; then + # Build and install collection + ansible-galaxy collection build + ansible-galaxy collection install kubernetes_sigs-kubespray-$(grep "^version:" galaxy.yml | awk '{print $2}').tar.gz + + # Simply remove all of our files and directories except for our tests directory + # to be absolutely certain that none of our playbooks or roles + # are interfering with our collection + find -maxdepth 1 ! -name tests -exec rm -rfv {} \; + + # Write cluster.yml +cat > cluster.yml < reset.yml < remove-node.yml <, expected: <$1>." + fi +} + +assertion__different() { + if [[ "$1" == "$2" ]]; then + _assertion__failed "Both values are: <$1>." + fi +} + +assertion__string_contains() { + if ! system__string_contains "$1" "$2"; then + _assertion__failed "String: <$1> does not contain: <$2>." + fi +} + +assertion__string_does_not_contain() { + if system__string_contains "$1" "$2"; then + _assertion__failed "String: <$1> contains: <$2>." + fi +} + +assertion__string_empty() { + if [[ -n "$1" ]]; then + _assertion__failed "String: <$1> is not empty." + fi +} + +assertion__string_not_empty() { + if [[ -z "$1" ]]; then + _assertion__failed "The string is empty." + fi +} + +assertion__array_contains() { + local element=$1 + shift 1 + if ! array__contains "${element}" "$@"; then + local array_as_string="$(system__pretty_print_array "$@")" + _assertion__failed \ + "Array: <${array_as_string}> does not contain: <${element}>." + fi +} + +assertion__array_does_not_contain() { + local element=$1 + shift 1 + if array__contains "${element}" "$@"; then + local array_as_string="$(system__pretty_print_array "$@")" + _assertion__failed \ + "Array: <${array_as_string}> contains: <${element}>." + fi +} + +assertion__status_code_is_success() { + if (( $1 != ${SBU_SUCCESS_STATUS_CODE} )); then + _assertion__failed \ + "Status code is failure instead of success." "$2" + fi +} + +assertion__status_code_is_failure() { + if (( $1 == ${SBU_SUCCESS_STATUS_CODE} )); then + _assertion__failed \ + "Status code is success instead of failure." "$2" + fi +} + +assertion__successful() { + "$@" + if (( $? != ${SBU_SUCCESS_STATUS_CODE} )); then + _assertion__failed "Command is failing instead of successful." + fi +} + +assertion__failing() { + "$@" + if (( $? == ${SBU_SUCCESS_STATUS_CODE} )); then + _assertion__failed "Command is successful instead of failing." + fi +} + +_assertion__failed() { + local message_to_use="$(_assertion__get_assertion_message_to_use "$1" "$2")" + system__print_line "Assertion failed. ${message_to_use}" + exit ${SBU_FAILURE_STATUS_CODE} +} + +_assertion__get_assertion_message_to_use() { + local message=$1 + local custom_messsage=$2 + if [[ -n "${custom_messsage}" ]]; then + system__print "${message} ${custom_messsage}" + else + system__print "${message}" + fi +} + + +mock__make_function_do_nothing() { + mock__make_function_call "$1" ":" +} + +mock__make_function_prints() { + local function=$1 + local text=$2 + eval "${function}() { printf "${text}"; }" +} + +mock__make_function_call() { + local function_to_mock=$1 + local function_to_call=$2 + shift 2 + eval "${function_to_mock}() { ${function_to_call} \"\$@\"; }" +} + + +runner__run_all_test_files() { + SBU_BASE_TEST_DIRECTORY=$1 + reporter__test_files_start_running + timer__store_current_time "global_time" + results__test_files_start_running + _runner__run_all_test_files_with_pattern_in_directory "$1" + reporter__test_files_end_running "$(timer__get_time_elapsed "global_time")" + runner__tests_are_successful +} + +_runner__run_all_test_files_with_pattern_in_directory() { + local file + local files + array__from_lines files <<< "$(_runner__get_test_files_in_directory "$1")" + for file in "${files[@]}"; do + file_runner__run_test_file "${file}" + done +} + +_runner__get_test_files_in_directory() { + local files + array__from_lines files <<< "$(find "$1" -name "${SBU_TEST_FILE_PATTERN}" | sort)" + if [[ "${SBU_RANDOM_RUN}" == "${SBU_YES}" ]]; then + array__from_lines files <<< "$(system__randomize_array "${files[@]}")" + fi + array__print "${files[@]}" +} + +runner__tests_are_successful() { + (( $(results__get_failing_tests_count) == 0 \ + && $(results__get_skipped_tests_count) == 0 )) +} + + +file_runner__run_test_file() { + local file=$1 + local public_functions=($(parser__get_public_functions_in_file "${file}")) + local test_functions=($(_file_runner__get_test_functions)) + reporter__test_file_starts_running "${file}" "${#test_functions[@]}" + ( source "${file}" + _file_runner__run_global_setup_if_exists \ + && _file_runner__call_all_tests + _file_runner__run_global_teardown_if_exists ) + _file_runner__check_if_global_setup_has_exited + reporter__test_file_ends_running +} + +_file_runner__run_all_tests_if_global_setup_is_successful() { + _file_runner__call_all_tests +} + +_file_runner__call_all_tests() { + local i + for (( i=0; i < ${#test_functions[@]}; i++ )); do + test_runner__run_test "${test_functions[${i}]}" "${public_functions[@]}" + done +} + +_file_runner__skip_all_tests() { + local i + for (( i=0; i < ${#test_functions[@]}; i++ )); do + test_runner__skip_test "${test_functions[${i}]}" "${public_functions[@]}" + done +} + +_file_runner__get_test_functions() { + local result=() + local test_function + for test_function in "${public_functions[@]}"; do + if _file_runner__function_is_a_test "${test_function}"\ + && [[ "${test_function}" == ${SBU_TEST_FUNCTION_PATTERN} ]]; then + result+=("${test_function}") + fi + done + _file_runner__get_randomized_test_functions_if_needed "${result[@]}" +} + +_file_runner__get_randomized_test_functions_if_needed() { + if [[ "${SBU_RANDOM_RUN}" == "${SBU_YES}" ]]; then + system__randomize_array "$@" + else + array__print "$@" + fi +} + +_file_runner__run_global_setup_if_exists() { + database__put "sbu_current_global_setup_has_failed" "${SBU_YES}" + _file_runner__call_function_if_exists "${SBU_GLOBAL_SETUP_FUNCTION_NAME}" \ + && database__put "sbu_current_global_setup_has_failed" "${SBU_NO}" +} + +_file_runner__run_global_teardown_if_exists() { + _file_runner__call_function_if_exists "${SBU_GLOBAL_TEARDOWN_FUNCTION_NAME}" +} + +_file_runner__function_is_a_test() { + ! array__contains "$1" \ + "${SBU_GLOBAL_SETUP_FUNCTION_NAME}" \ + "${SBU_GLOBAL_TEARDOWN_FUNCTION_NAME}" \ + "${SBU_SETUP_FUNCTION_NAME}" \ + "${SBU_TEARDOWN_FUNCTION_NAME}" +} + +_file_runner__call_function_if_exists() { + local function=$1 + shift 1 + if array__contains "${function}" "${public_functions[@]}"; then + "${function}" + fi +} + +_file_runner__check_if_global_setup_has_exited() { + local has_exited="$(database__get "sbu_current_global_setup_has_failed")" + if [[ "${has_exited}" == "${SBU_YES}" ]]; then + _file_runner__handle_failure_in_global_setup + fi +} + +_file_runner__handle_failure_in_global_setup() { + reporter__global_setup_has_failed + _file_runner__skip_all_tests +} + + +parser__get_public_functions_in_file() { + _parser__find_functions_in_file "$1" \ + | _parser__filter_private_functions \ + | awk '{ print $1 }' +} + +_parser__find_functions_in_file() { + grep -o "${SBU_FUNCTION_DECLARATION_REGEX}" "$1" \ + | _parser__get_function_name_from_declaration +} + +_parser__filter_private_functions() { + grep -v "${SBU_PRIVATE_FUNCTION_NAME_REGEX}" +} + +_parser__get_function_name_from_declaration() { + sed "s/${SBU_FUNCTION_DECLARATION_REGEX}/\2/" +} + + +timer__store_current_time() { + local id=$1 + database__put "sbu_beginning_date_$1" "$(system__get_date_in_seconds)" +} + +timer__get_time_elapsed() { + local id=$1 + local beginning_date="$(database__get "sbu_beginning_date_$1")" + local ending_date="$(system__get_date_in_seconds)" + + [[ -n "${beginning_date}" ]] \ + && system__print "$(( ending_date - beginning_date ))" \ + || system__print "0" +} + + +results__test_files_start_running() { + database__put "sbu_successful_tests_count" "0" + database__put "sbu_failing_tests_count" "0" + database__put "sbu_skipped_tests_count" "0" +} + +results__get_successful_tests_count() { + _results__get_tests_count_of_type "successful" +} + +results__increment_successful_tests() { + _results__increment_tests_of_type "successful" +} + +results__get_failing_tests_count() { + _results__get_tests_count_of_type "failing" +} + +results__increment_failing_tests() { + _results__increment_tests_of_type "failing" +} + +results__get_skipped_tests_count() { + _results__get_tests_count_of_type "skipped" +} + +results__increment_skipped_tests() { + _results__increment_tests_of_type "skipped" +} + +results__get_total_tests_count() { + local successes="$(results__get_successful_tests_count)" + local failures="$(results__get_failing_tests_count)" + local skipped="$(results__get_skipped_tests_count)" + system__print "$(( successes + failures + skipped ))" +} + +_results__get_tests_count_of_type() { + local type=$1 + database__get "sbu_${type}_tests_count" +} + +_results__increment_tests_of_type() { + local type=$1 + local count="$(results__get_${type}_tests_count)" + database__put "sbu_${type}_tests_count" "$(( count + 1 ))" +} + + +test_runner__run_test() { + local test_function=$1 + shift 1 + reporter__test_starts_running "${test_function}" + timer__store_current_time "test_time" + ( + _test_runner__call_setup_if_exists "$@" \ + && _test_runner__call_test_fonction "${test_function}" + local setup_and_test_code=$? + _test_runner__call_teardown_if_exists "$@" + (( $? == ${SBU_SUCCESS_STATUS_CODE} \ + && ${setup_and_test_code} == ${SBU_SUCCESS_STATUS_CODE} )) + ) + _test_runner__parse_test_function_result $? + reporter__test_ends_running "$(timer__get_time_elapsed "test_time")" +} + +_test_runner__call_test_fonction() { + ( "$1" >&${SBU_STANDARD_FD} 2>&${SBU_ERROR_FD} ) +} + +_test_runner__call_setup_if_exists() { + _test_runner__call_function_if_exits "${SBU_SETUP_FUNCTION_NAME}" "$@" +} + +_test_runner__call_teardown_if_exists() { + _test_runner__call_function_if_exits "${SBU_TEARDOWN_FUNCTION_NAME}" "$@" +} + +_test_runner__parse_test_function_result() { + if (( $1 == ${SBU_SUCCESS_STATUS_CODE} )); then + results__increment_successful_tests + reporter__test_has_succeeded + else + results__increment_failing_tests + reporter__test_has_failed + fi +} + +_test_runner__call_function_if_exits() { + local function=$1 + shift 1 + if array__contains "${function}" "$@"; then + "${function}" + fi +} + +test_runner__skip_test() { + local test_function=$1 + reporter__test_starts_running "${test_function}" + results__increment_skipped_tests + reporter__test_is_skipped "${test_function}" + reporter__test_ends_running 0 +} + + +reporter__test_files_start_running() { + _reporter__initialise_file_descriptors + reporter__for_each_reporter \ + _reporter__call_function "test_files_start_running" "$@" +} + +_reporter__initialise_file_descriptors() { + eval "exec ${SBU_STANDARD_FD}>&1" + eval "exec ${SBU_ERROR_FD}>&2" +} + +reporter__global_setup_has_failed() { + reporter__for_each_reporter \ + _reporter__call_function "global_setup_has_failed" "$@" +} + +reporter__test_file_starts_running() { + reporter__for_each_reporter \ + _reporter__call_function "test_file_starts_running" "$@" +} + +reporter__test_starts_running() { + reporter__for_each_reporter \ + _reporter__call_function "test_starts_running" "$@" +} + +reporter__test_has_succeeded() { + reporter__for_each_reporter \ + _reporter__call_function "test_has_succeeded" "$@" +} + +reporter__test_has_failed() { + reporter__for_each_reporter \ + _reporter__call_function "test_has_failed" "$@" +} + +reporter__test_is_skipped() { + reporter__for_each_reporter \ + _reporter__call_function "test_is_skipped" "$@" +} + +reporter__test_ends_running() { + reporter__for_each_reporter \ + _reporter__call_function "test_ends_running" "$@" +} + +reporter__test_file_ends_running() { + reporter__for_each_reporter \ + _reporter__call_function "test_file_ends_running" "$@" +} + +reporter__test_files_end_running() { + reporter__for_each_reporter \ + _reporter__call_function "test_files_end_running" "$@" + _reporter__release_file_descriptors +} + +_reporter__release_file_descriptors() { + eval "exec 1>&${SBU_STANDARD_FD} ${SBU_STANDARD_FD}>&-" + eval "exec 2>&${SBU_ERROR_FD} ${SBU_ERROR_FD}>&-" +} + +_reporter__call_function() { + local function=$1 + shift 1 + "${reporter}_reporter__${function}" "$@" +} + +reporter__for_each_reporter() { + local reporter + for reporter in ${SBU_REPORTERS//${SBU_VALUE_SEPARATOR}/ }; do + "$@" + done +} + +reporter__print_with_color() { + system__print_with_color "$@" >&${SBU_STANDARD_FD} +} + +reporter__print_line() { + system__print_line "$@" >&${SBU_STANDARD_FD} +} + +reporter__print_line_with_color() { + system__print_line_with_color "$@" >&${SBU_STANDARD_FD} +} + +reporter__print_new_line() { + system__print_new_line >&${SBU_STANDARD_FD} +} + +reporter__get_color_code_for_tests_result() { + local color_code=${SBU_GREEN_COLOR_CODE} + if ! runner__tests_are_successful; then + color_code=${SBU_RED_COLOR_CODE} + fi + system__print "${color_code}" +} + +reporter__get_test_file_relative_name() { + system__print "${1#${SBU_BASE_TEST_DIRECTORY}\/}" +} + + +simple_reporter__test_files_start_running() { + : +} + +simple_reporter__test_file_starts_running() { + local relative_name="$(reporter__get_test_file_relative_name "$1")" + reporter__print_line "[File] ${relative_name}" +} + +simple_reporter__global_setup_has_failed() { + reporter__print_line_with_color \ + "Global setup has failed" ${SBU_YELLOW_COLOR_CODE} +} + +simple_reporter__test_starts_running() { + reporter__print_line "[Test] $1" +} + +simple_reporter__test_has_succeeded() { + reporter__print_line_with_color "OK" ${SBU_GREEN_COLOR_CODE} +} + +simple_reporter__test_has_failed() { + reporter__print_line_with_color "KO" ${SBU_RED_COLOR_CODE} +} + +simple_reporter__test_is_skipped() { + reporter__print_line_with_color "Skipped" ${SBU_YELLOW_COLOR_CODE} +} + +simple_reporter__test_ends_running() { + : +} + +simple_reporter__test_file_ends_running() { + reporter__print_new_line +} + +simple_reporter__test_files_end_running() { + local time="in $1s" + reporter__print_line "[Results]" + local color="$(reporter__get_color_code_for_tests_result)" + local total_count="$(_simple_reporter__get_total_count_message)" + local failures_count="$(_simple_reporter__get_failures_count_message)" + local skipped_count="$(results__get_skipped_tests_count) skipped" + local message="${total_count}, ${failures_count}, ${skipped_count} ${time}" + reporter__print_line_with_color "${message}" "${color}" +} + +_simple_reporter__get_total_count_message() { + local count="$(results__get_total_tests_count)" + system__print "${count} test$(_simple_reporter__get_agreement ${count})" +} + +_simple_reporter__get_failures_count_message() { + local count="$(results__get_failing_tests_count)" + system__print "${count} failure$(_simple_reporter__get_agreement ${count})" +} + +_simple_reporter__get_agreement() { + (( $1 > 1 )) \ + && system__print "s" \ + || system__print "" +} + + +dots_reporter__test_files_start_running() { + exec 1>/dev/null + exec 2>/dev/null +} + +dots_reporter__test_file_starts_running() { + : +} + +dots_reporter__global_setup_has_failed() { + : +} + +dots_reporter__test_starts_running() { + : +} + +dots_reporter__test_has_succeeded() { + reporter__print_with_color "." ${SBU_GREEN_COLOR_CODE} +} + +dots_reporter__test_has_failed() { + reporter__print_with_color "F" ${SBU_RED_COLOR_CODE} +} + +dots_reporter__test_is_skipped() { + reporter__print_with_color "S" ${SBU_YELLOW_COLOR_CODE} +} + +dots_reporter__test_ends_running() { + : +} + +dots_reporter__test_file_ends_running() { + : +} + +dots_reporter__test_files_end_running() { + local color="$(reporter__get_color_code_for_tests_result)" + local texte="$(runner__tests_are_successful \ + && system__print "OK" \ + || system__print "KO")" + reporter__print_line_with_color "${texte}" "${color}" +} + + +junit_reporter__test_files_start_running() { + _junit_reporter__initialise_report_with \ + "" + _junit_reporter__write_line_to_report "" +} + +junit_reporter__test_file_starts_running() { + local file_name=$1 + local test_count=$2 + local suite_name="$(_junit_reporter__get_suite_name "${file_name}")" + database__put "sbu_current_suite_name" "${suite_name}" + _junit_reporter__write_line_to_report \ + " " + _junit_reporter__delete_all_outputs_lines "suite" + _junit_reporter__redirect_outputs_to_database "suite" +} + +junit_reporter__global_setup_has_failed() { + : +} + +junit_reporter__test_starts_running() { + local suite_name="$(database__get "sbu_current_suite_name")" + local test_name="$(xml__encode_text "$1")" + _junit_reporter__write_line_to_report \ + " " + _junit_reporter__delete_all_outputs_lines "test" + _junit_reporter__redirect_outputs_to_database "test" +} + +junit_reporter__test_has_succeeded() { + : +} + +junit_reporter__test_has_failed() { + _junit_reporter__write_line_to_report " " + _junit_reporter__write_line_to_report " " +} + +junit_reporter__test_is_skipped() { + _junit_reporter__write_line_to_report " " + _junit_reporter__write_line_to_report " " +} + +junit_reporter__test_ends_running() { + _junit_reporter__redirect_outputs_to_database "suite" + _junit_reporter__write_time_in_current_test_case_tag_in_report "$1" + _junit_reporter__flush_all_outputs_to_report_if_any "test" + _junit_reporter__write_line_to_report " " +} + +_junit_reporter__write_time_in_current_test_case_tag_in_report() { + local test_time=$1 + local report_content=$(cat "${SBU_JUNIT_REPORTER_OUTPUT_FILE}") + local content_with_time="$(system__substitute_variable \ + "${report_content}" "sbu_current_test_time" "${test_time}")" + system__print_line \ + "${content_with_time}" > "${SBU_JUNIT_REPORTER_OUTPUT_FILE}" +} + +junit_reporter__test_file_ends_running() { + _junit_reporter__flush_all_outputs_to_report_if_any "suite" + _junit_reporter__write_line_to_report " " + database__put "sbu_current_suite_name" "" +} + +junit_reporter__test_files_end_running() { + _junit_reporter__write_line_to_report "" +} + +_junit_reporter__get_suite_name() { + local relative_name="$(reporter__get_test_file_relative_name "$1")" + local dots_replaced_by_underscores="${relative_name//./_}" + local slashes_replaced_by_dots="${dots_replaced_by_underscores//\//.}" + xml__encode_text "${slashes_replaced_by_dots}" +} + +_junit_reporter__initialise_report_with() { + system__print_line "$1" > "${SBU_JUNIT_REPORTER_OUTPUT_FILE}" +} + +_junit_reporter__write_line_to_report() { + system__print_line "$1" >> "${SBU_JUNIT_REPORTER_OUTPUT_FILE}" +} + +_junit_reporter__redirect_outputs_to_database() { + local scope=$1 + exec 1>>\ + "$(database__get_descriptor "sbu_current_${scope}_standard_ouputs_lines")" + exec 2>>\ + "$(database__get_descriptor "sbu_current_${scope}_error_ouputs_lines")" +} + +_junit_reporter__delete_all_outputs_lines() { + database__put "sbu_current_$1_standard_ouputs_lines" + database__put "sbu_current_$1_error_ouputs_lines" +} + +_junit_reporter__flush_all_outputs_to_report_if_any() { + _junit_reporter__flush_outputs_to_report_if_any "$1" "standard" + _junit_reporter__flush_outputs_to_report_if_any "$1" "error" +} + +_junit_reporter__flush_outputs_to_report_if_any() { + local scope=$1 + local outputs_type=$2 + local key="sbu_current_${scope}_${outputs_type}_ouputs_lines" + local outputs="$(database__get "${key}")" + if [[ -n "${outputs}" ]]; then + _junit_reporter__write_outputs_to_report \ + "${scope}" "${outputs_type}" "${outputs}" + database__put "${key}" "" + fi +} + +_junit_reporter__write_outputs_to_report() { + local scope=$1 + local outputs_type=$2 + local outputs=$3 + local tag="$(_junit_reporter__get_tag_for_outputs_type "${outputs_type}")" + local indentation="$(_junit_reporter__get_indentation_for_scope "${scope}")" + _junit_reporter__write_line_to_report "${indentation}<${tag}>" + _junit_reporter__write_line_to_report "$(xml__encode_text "${outputs}")" + _junit_reporter__write_line_to_report "${indentation}" +} + +_junit_reporter__get_tag_for_outputs_type() { + [[ "$1" == "standard" ]] \ + && system__print "system-out" \ + || system__print "system-err" +} + +_junit_reporter__get_indentation_for_scope() { + [[ "$1" == "suite" ]] \ + && system__print " " \ + || system__print " " +} + + +xml__encode_text() { + local encoded=${1//\&/\&\;} + encoded=${encoded//\/\>\;} + encoded=${encoded//\"/\"\;} + encoded=${encoded//\'/\&apos\;} + system__print "${encoded}" +} + + +database__initialise() { + _SBU_DB_TOKEN="$(system__random)" + _database__ensure_directory_exists +} + +database__release() { + rm -rf "$(_database__get_dir)" +} + +database__put() { + _database__ensure_directory_exists + system__print "$2" > "$(_database__get_dir)/$1" +} + +database__post() { + _database__ensure_directory_exists + system__print "$2" >> "$(_database__get_dir)/$1" +} + +database__post_line() { + _database__ensure_directory_exists + system__print_line "$2" >> "$(_database__get_dir)/$1" +} + +database__put_variable() { + _database__ensure_directory_exists + database__put "$1" "${!1}" +} + +database__get() { + [[ -e "$(_database__get_dir)/$1" ]] && cat "$(_database__get_dir)/$1" +} + +database__get_descriptor() { + system__print "$(_database__get_dir)/$1" +} + +_database__ensure_directory_exists() { + mkdir -p "$(_database__get_dir)" +} + +_database__get_dir() { + system__print "${SBU_TEMP_DIR}/database/${_SBU_DB_TOKEN}" +} + + +system__get_string_or_default() { + [[ -n "$1" ]] \ + && system__print "$1" \ + || system__print "$2" +} + +system__get_date_in_seconds() { + date +%s +} + +system__print_line_with_color() { + system__print_with_color "$@" + system__print_new_line +} + +system__print_with_color() { + if [[ "${SBU_USE_COLORS}" == "${SBU_YES}" ]]; then + printf "$2$1${SBU_DEFAULT_COLOR_CODE}" + else + system__print "$1" + fi +} + +system__print_line() { + system__print "$1" + system__print_new_line +} + +system__print() { + printf "%s" "$1" +} + +system__print_new_line() { + printf "\n" +} + +array__contains() { + local value=$1 + shift 1 + local i + for (( i=1; i <= $#; i++ )); do + if [[ "${!i}" == "${value}" ]]; then + return ${SBU_SUCCESS_STATUS_CODE} + fi + done + return ${SBU_FAILURE_STATUS_CODE} +} + +array__from_lines() { + local IFS=$'\n' + eval "$1=(\$( 0 )); do + local random_index=$(( $(system__random) % ${#copy[@]} )) + system__print_line "${copy[${random_index}]}" + unset copy[${random_index}] + copy=("${copy[@]}") + done +} + +system__random() { + system__print "${RANDOM}" +} + +system__substitute_variable() { + local string=$1 + local key="\$\{$2\}" + local value=$3 + printf "%s" "${string//${key}/${value}}" +} + + +main__main() { + configuration__load + _main__initialise + local parsed_arguments=0 + _main__parse_arguments "$@" + shift ${parsed_arguments} + _main__assert_only_one_argument_left $# + _main__assert_reporters_are_known + SBU_BASE_TEST_DIRECTORY=$1 + + if [[ "${SBU_NO_RUN}" != "${SBU_YES}" ]]; then + runner__run_all_test_files "$1" + return $? + fi +} + +_main__initialise() { + database__initialise + trap _main__release EXIT +} + +_main__release() { + database__release +} + +_main__parse_arguments() { + local argument + for argument in "$@"; do + case "${argument}" in + -a|--api-cheat-sheet) + _main__print_api_cheat_sheet_and_exit + ;; + -c=*|--colors=*) + SBU_USE_COLORS="${argument#*=}" + (( parsed_arguments++ )) + ;; + -d=*|--random-run=*) + SBU_RANDOM_RUN="${argument#*=}" + (( parsed_arguments++ )) + ;; + -h|--help) + _main__print_full_usage + exit ${SBU_SUCCESS_STATUS_CODE} + ;; + -f=*|--file-pattern=*) + SBU_TEST_FILE_PATTERN="${argument#*=}" + (( parsed_arguments++ )) + ;; + --no-run) + SBU_NO_RUN="${SBU_YES}" + (( parsed_arguments++ )) + ;; + -o=*|--output-file=*) + SBU_JUNIT_REPORTER_OUTPUT_FILE="${argument#*=}" + (( parsed_arguments++ )) + ;; + -t=*|--test-pattern=*) + SBU_TEST_FUNCTION_PATTERN="${argument#*=}" + (( parsed_arguments++ )) + ;; + -r=*|--reporters=*) + SBU_REPORTERS="${argument#*=}" + (( parsed_arguments++ )) + ;; + -*|--*) + _main__print_illegal_option "${argument}" + _main__print_usage_and_exit_with_code ${SBU_FAILURE_STATUS_CODE} + ;; + esac + done +} + + _main__assert_reporters_are_known() { + reporter__for_each_reporter _main__fail_if_reporter_unknown +} + +_main__fail_if_reporter_unknown() { + if ! array__contains "${reporter}" "simple" "dots" "junit"; then + system__print_line \ + "$(_main__get_script_name): unknown reporter <${reporter}>" + exit ${SBU_FAILURE_STATUS_CODE} + fi +} + +_main__print_illegal_option() { + local option="${1%=*}" + option="${option#-}" + option="${option#-}" + system__print_line "$(_main__get_script_name): illegal option -- ${option}" +} + +_main__assert_only_one_argument_left() { + if (( $1 > 1 )); then + system__print_line "$(_main__get_script_name): only one path is allowed" + _main__print_usage_and_exit_with_code ${SBU_FAILURE_STATUS_CODE} + fi +} + +_main__get_script_name() { + basename "${BASH_SOURCE[0]}" +} + +_main__print_usage_and_exit_with_code() { + _main__print_usage + exit $1 +} + +_main__print_full_usage() { + _main__print_usage + local script="$(_main__get_script_name)" + system__print_new_line + system__print_line "\ +[options] + -a, --api-cheat-sheet + print api cheat sheet like assertions + -c, --colors=${SBU_YES} or ${SBU_NO} + tests output with colors or no + -d, --random-run=${SBU_YES} or ${SBU_NO} + tests files and functions randomly run or no + -f, --file-pattern= + pattern to filter test files + -h + print usage + -o, --output-file= + output file for JUnit reporter + -r, --reporters= + comma-separated reporters (simple, dots or junit) + -t, --test-pattern= + pattern to filter test function in files + +[examples] + ${script} . + run all tests in current directory + ${script} -p=*test.sh sources/test + run all tests files ending with test.sh in sources/test" +} + +_main__print_usage() { + system__print_line "\ +usage: $(_main__get_script_name) [options] path + run all tests in path" +} + +_main__print_api_cheat_sheet_and_exit() { + system__print_line "\ +[assertions] + assertion__equal (value, other) + -> assert that is equal to + assertion__different (value, other) + -> assert that is different from + assertion__string_contains (string, substring) + -> assert that contains + assertion__string_does_not_contain (string, substring) + -> assert that does not contain + assertion__string_empty (string) + -> assert that is empty + assertion__string_not_empty (string) + -> assert that is not empty + assertion__array_contains (element, array[0], array[1], ...) + -> assert that the contains the + assertion__array_does_not_contain (element, array elements...) + -> assert that the does not contain the + assertion__successful (command) + -> assert that the is successful + assertion__failing (command) + -> assert that the is failing + assertion__status_code_is_success (code) + -> assert that the status is 0 + assertion__status_code_is_failure (code) + -> assert that the status is not 0 + +[special functions] + ${SBU_GLOBAL_SETUP_FUNCTION_NAME} + -> Executed before all tests in a file + ${SBU_GLOBAL_TEARDOWN_FUNCTION_NAME} + -> Executed after all tests in a file + ${SBU_SETUP_FUNCTION_NAME} + -> Executed before each test in a file + ${SBU_TEARDOWN_FUNCTION_NAME} + -> Executed after each test in a file + +[mocks] + mock__make_function_do_nothing (function_to_mock) + -> make function do nothing + mock__make_function_prints (function_to_mock, message) + -> make function prints a message + mock__make_function_call (function_to_mock, function_to_call) + -> make function call another function" + exit ${SBU_SUCCESS_STATUS_CODE} +} + + +main__main "$@" diff --git a/kubespray/tests/support/aws.groovy b/kubespray/tests/support/aws.groovy new file mode 100644 index 0000000..bc13b51 --- /dev/null +++ b/kubespray/tests/support/aws.groovy @@ -0,0 +1,94 @@ +def run(username, credentialsId, ami, network_plugin, aws_access, aws_secret) { + def inventory_path = pwd() + "/inventory/sample/${env.CI_JOB_NAME}-${env.BUILD_NUMBER}.ini" + dir('tests') { + wrap([$class: 'AnsiColorBuildWrapper', colorMapName: "xterm"]) { + try { + create_vm("${env.CI_JOB_NAME}-${env.BUILD_NUMBER}", inventory_path, ami, username, network_plugin, aws_access, aws_secret) + install_cluster(inventory_path, credentialsId, network_plugin) + + test_apiserver(inventory_path, credentialsId) + test_create_pod(inventory_path, credentialsId) + test_network(inventory_path, credentialsId) + } finally { + delete_vm(inventory_path, credentialsId, aws_access, aws_secret) + } + } + } +} + +def create_vm(run_id, inventory_path, ami, username, network_plugin, aws_access, aws_secret) { + ansiblePlaybook( + inventory: 'local_inventory/hosts.cfg', + playbook: 'cloud_playbooks/create-aws.yml', + extraVars: [ + test_id: run_id, + kube_network_plugin: network_plugin, + aws_access_key: [value: aws_access, hidden: true], + aws_secret_key: [value: aws_secret, hidden: true], + aws_ami_id: ami, + aws_security_group: [value: 'sg-cb0327a2', hidden: true], + key_name: 'travis-ci', + inventory_path: inventory_path, + aws_region: 'eu-central-1', + ssh_user: username + ], + colorized: true + ) +} + +def delete_vm(inventory_path, credentialsId, aws_access, aws_secret) { + ansiblePlaybook( + inventory: inventory_path, + playbook: 'cloud_playbooks/delete-aws.yml', + credentialsId: credentialsId, + extraVars: [ + aws_access_key: [value: aws_access, hidden: true], + aws_secret_key: [value: aws_secret, hidden: true] + ], + colorized: true + ) +} + +def install_cluster(inventory_path, credentialsId, network_plugin) { + ansiblePlaybook( + inventory: inventory_path, + playbook: '../cluster.yml', + sudo: true, + credentialsId: credentialsId, + extraVars: [ + kube_network_plugin: network_plugin + ], + extras: "-e cloud_provider=aws", + colorized: true + ) +} + +def test_apiserver(inventory_path, credentialsId) { + ansiblePlaybook( + inventory: inventory_path, + playbook: 'testcases/010_check-apiserver.yml', + credentialsId: credentialsId, + colorized: true + ) +} + +def test_create_pod(inventory_path, credentialsId) { + ansiblePlaybook( + inventory: inventory_path, + playbook: 'testcases/020_check-create-pod.yml', + sudo: true, + credentialsId: credentialsId, + colorized: true + ) +} + +def test_network(inventory_path, credentialsId) { + ansiblePlaybook( + inventory: inventory_path, + playbook: 'testcases/030_check-network.yml', + sudo: true, + credentialsId: credentialsId, + colorized: true + ) +} +return this; diff --git a/kubespray/tests/templates/fake_hosts.yml.j2 b/kubespray/tests/templates/fake_hosts.yml.j2 new file mode 100644 index 0000000..c172b78 --- /dev/null +++ b/kubespray/tests/templates/fake_hosts.yml.j2 @@ -0,0 +1,3 @@ +ansible_default_ipv4: + address: 255.255.255.255 +ansible_hostname: "{{ '{{' }}inventory_hostname }}" diff --git a/kubespray/tests/templates/inventory-aws.j2 b/kubespray/tests/templates/inventory-aws.j2 new file mode 100644 index 0000000..e3c5373 --- /dev/null +++ b/kubespray/tests/templates/inventory-aws.j2 @@ -0,0 +1,29 @@ +node1 ansible_ssh_host={{ec2.instances[0].public_ip}} ansible_ssh_user={{ssh_user}} +node2 ansible_ssh_host={{ec2.instances[1].public_ip}} ansible_ssh_user={{ssh_user}} +node3 ansible_ssh_host={{ec2.instances[2].public_ip}} ansible_ssh_user={{ssh_user}} + +[kube_control_plane] +node1 +node2 + +[kube_node] +node1 +node2 +node3 + +[etcd] +node1 +node2 + +[k8s_cluster:children] +kube_node +kube_control_plane +calico_rr + +[calico_rr] + +[broken_kube_control_plane] +node2 + +[broken_etcd] +node2 diff --git a/kubespray/tests/templates/inventory-do.j2 b/kubespray/tests/templates/inventory-do.j2 new file mode 100644 index 0000000..fb54361 --- /dev/null +++ b/kubespray/tests/templates/inventory-do.j2 @@ -0,0 +1,47 @@ +{% for instance in droplets.results %} +{{instance.droplet.name}} ansible_ssh_host={{instance.droplet.ip_address}} +{% endfor %} + +{% if mode is defined and mode == "separate" %} +[kube_control_plane] +{{droplets.results[0].droplet.name}} + +[kube_node] +{{droplets.results[1].droplet.name}} + +[etcd] +{{droplets.results[2].droplet.name}} +{% elif mode is defined and mode == "ha" %} +[kube_control_plane] +{{droplets.results[0].droplet.name}} +{{droplets.results[1].droplet.name}} + +[kube_node] +{{droplets.results[2].droplet.name}} + +[etcd] +{{droplets.results[1].droplet.name}} +{{droplets.results[2].droplet.name}} + +[broken_kube_control_plane] +{{droplets.results[1].droplet.name}} + +[broken_etcd] +{{droplets.results[2].droplet.name}} +{% else %} +[kube_control_plane] +{{droplets.results[0].droplet.name}} + +[kube_node] +{{droplets.results[1].droplet.name}} + +[etcd] +{{droplets.results[0].droplet.name}} +{% endif %} + +[calico_rr] + +[k8s_cluster:children] +kube_node +kube_control_plane +calico_rr diff --git a/kubespray/tests/templates/inventory-gce.j2 b/kubespray/tests/templates/inventory-gce.j2 new file mode 100644 index 0000000..33e9bbc --- /dev/null +++ b/kubespray/tests/templates/inventory-gce.j2 @@ -0,0 +1,73 @@ +{% set node1 = gce.instance_data[0].name %} +{{node1}} ansible_ssh_host={{gce.instance_data[0].public_ip}} +{% if mode != "aio" %} +{% set node2 = gce.instance_data[1].name %} +{{node2}} ansible_ssh_host={{gce.instance_data[1].public_ip}} +{% endif %} +{% if mode is defined and mode in ["ha", "ha-scale", "separate", "separate-scale"] %} +{% set node3 = gce.instance_data[2].name %} +{{node3}} ansible_ssh_host={{gce.instance_data[2].public_ip}} +{% endif %} +{% if mode is defined and mode in ["separate", "separate-scale"] %} +[kube_control_plane] +{{node1}} + +[kube_node] +{{node2}} + +[etcd] +{{node3}} + +{% elif mode is defined and mode in ["ha", "ha-scale"] %} +[kube_control_plane] +{{node1}} +{{node2}} + +[kube_node] +{{node3}} + +[etcd] +{{node1}} +{{node2}} +{{node3}} + +[broken_kube_control_plane] +{{node2}} + +[etcd] +{{node2}} +{{node3}} +{% elif mode == "default" %} +[kube_control_plane] +{{node1}} + +[kube_node] +{{node2}} + +[etcd] +{{node1}} +{% elif mode == "aio" %} +[kube_control_plane] +{{node1}} + +[kube_node] +{{node1}} + +[etcd] +{{node1}} +{% endif %} + +[k8s_cluster:children] +kube_node +kube_control_plane +calico_rr + +[calico_rr] + +{% if mode is defined and mode in ["scale", "separate-scale", "ha-scale"] %} +[fake_hosts] +fake_scale_host[1:200] + +[kube_node:children] +fake_hosts +{% endif %} diff --git a/kubespray/tests/testcases/010_check-apiserver.yml b/kubespray/tests/testcases/010_check-apiserver.yml new file mode 100644 index 0000000..0d20bda --- /dev/null +++ b/kubespray/tests/testcases/010_check-apiserver.yml @@ -0,0 +1,24 @@ +--- +- name: Testcases for apiserver + hosts: kube_control_plane + + tasks: + - name: Check the API servers are responding + uri: + url: "https://{{ access_ip | default(ansible_default_ipv4.address) }}:{{ kube_apiserver_port | default(6443) }}/version" + validate_certs: no + status_code: 200 + register: apiserver_response + retries: 12 + delay: 5 + until: apiserver_response is success + + - debug: # noqa name[missing] + msg: "{{ apiserver_response.json }}" + + - name: Check API servers version + assert: + that: + - apiserver_response.json.gitVersion == kube_version + fail_msg: "apiserver version different than expected {{ kube_version }}" + when: kube_version is defined diff --git a/kubespray/tests/testcases/015_check-nodes-ready.yml b/kubespray/tests/testcases/015_check-nodes-ready.yml new file mode 100644 index 0000000..69945ca --- /dev/null +++ b/kubespray/tests/testcases/015_check-nodes-ready.yml @@ -0,0 +1,36 @@ +--- +- name: Testcases checking nodes + hosts: kube_control_plane[0] + tasks: + + - name: Force binaries directory for Flatcar Container Linux by Kinvolk + set_fact: + bin_dir: "/opt/bin" + when: ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + + - name: Force binaries directory for other hosts + set_fact: + bin_dir: "/usr/local/bin" + when: not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + + - import_role: # noqa name[missing] + name: cluster-dump + + - name: Check kubectl output + command: "{{ bin_dir }}/kubectl get nodes" + changed_when: false + register: get_nodes + no_log: true + + - debug: # noqa name[missing] + msg: "{{ get_nodes.stdout.split('\n') }}" + + - name: Check that all nodes are running and ready + command: "{{ bin_dir }}/kubectl get nodes --no-headers -o yaml" + changed_when: false + register: get_nodes_yaml + until: + # Check that all nodes are Status=Ready + - '(get_nodes_yaml.stdout | from_yaml)["items"] | map(attribute = "status.conditions") | map("items2dict", key_name="type", value_name="status") | map(attribute="Ready") | list | min' + retries: 30 + delay: 10 diff --git a/kubespray/tests/testcases/020_check-pods-running.yml b/kubespray/tests/testcases/020_check-pods-running.yml new file mode 100644 index 0000000..54f39ea --- /dev/null +++ b/kubespray/tests/testcases/020_check-pods-running.yml @@ -0,0 +1,50 @@ +--- +- name: Testcases checking pods + hosts: kube_control_plane[0] + tasks: + + - name: Force binaries directory for Flatcar Container Linux by Kinvolk + set_fact: + bin_dir: "/opt/bin" + when: ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + + - name: Force binaries directory for other hosts + set_fact: + bin_dir: "/usr/local/bin" + when: not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + + - import_role: # noqa name[missing] + name: cluster-dump + + - name: Check kubectl output + command: "{{ bin_dir }}/kubectl get pods --all-namespaces -owide" + changed_when: false + register: get_pods + no_log: true + + - debug: # noqa name[missing] + msg: "{{ get_pods.stdout.split('\n') }}" + + - name: Check that all pods are running and ready + command: "{{ bin_dir }}/kubectl get pods --all-namespaces --no-headers -o yaml" + changed_when: false + register: run_pods_log + until: + # Check that all pods are running + - '(run_pods_log.stdout | from_yaml)["items"] | map(attribute = "status.phase") | unique | list == ["Running"]' + # Check that all pods are ready + - '(run_pods_log.stdout | from_yaml)["items"] | map(attribute = "status.containerStatuses") | map("map", attribute = "ready") | map("min") | min' + retries: 30 + delay: 10 + failed_when: false + no_log: true + + - name: Check kubectl output + command: "{{ bin_dir }}/kubectl get pods --all-namespaces -owide" + changed_when: false + register: get_pods + no_log: true + + - debug: # noqa name[missing] + msg: "{{ get_pods.stdout.split('\n') }}" + failed_when: not run_pods_log is success diff --git a/kubespray/tests/testcases/030_check-network.yml b/kubespray/tests/testcases/030_check-network.yml new file mode 100644 index 0000000..ff63b5e --- /dev/null +++ b/kubespray/tests/testcases/030_check-network.yml @@ -0,0 +1,205 @@ +--- +- name: Testcases for network + hosts: kube_control_plane[0] + vars: + test_image_repo: registry.k8s.io/e2e-test-images/agnhost + test_image_tag: "2.40" + + tasks: + - name: Force binaries directory for Flatcar Container Linux by Kinvolk + set_fact: + bin_dir: "/opt/bin" + when: ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + + - name: Force binaries directory for other hosts + set_fact: + bin_dir: "/usr/local/bin" + when: not ansible_os_family in ["Flatcar", "Flatcar Container Linux by Kinvolk"] + + - name: Check kubelet serving certificates approved with kubelet_csr_approver + when: + - kubelet_rotate_server_certificates | default(false) + - kubelet_csr_approver_enabled | default(kubelet_rotate_server_certificates | default(false)) + block: + + - name: Get certificate signing requests + command: "{{ bin_dir }}/kubectl get csr" + register: get_csr + changed_when: false + + - debug: # noqa name[missing] + msg: "{{ get_csr.stdout.split('\n') }}" + + - name: Check there are csrs + assert: + that: get_csr.stdout_lines | length > 0 + fail_msg: kubelet_rotate_server_certificates is {{ kubelet_rotate_server_certificates }} but no csr's found + + - name: Get Denied/Pending certificate signing requests + shell: "set -o pipefail && {{ bin_dir }}/kubectl get csr | grep -e Denied -e Pending || true" + register: get_csr_denied_pending + changed_when: false + + - name: Check there are Denied/Pending csrs + assert: + that: get_csr_denied_pending.stdout_lines | length == 0 + fail_msg: kubelet_csr_approver is enabled but CSRs are not approved + + - name: Approve kubelet serving certificates + when: + - kubelet_rotate_server_certificates | default(false) + - not (kubelet_csr_approver_enabled | default(kubelet_rotate_server_certificates | default(false))) + block: + + - name: Get certificate signing requests + command: "{{ bin_dir }}/kubectl get csr -o name" + register: get_csr + changed_when: false + + - name: Check there are csrs + assert: + that: get_csr.stdout_lines | length > 0 + fail_msg: kubelet_rotate_server_certificates is {{ kubelet_rotate_server_certificates }} but no csr's found + + - name: Approve certificates + command: "{{ bin_dir }}/kubectl certificate approve {{ get_csr.stdout_lines | join(' ') }}" + register: certificate_approve + when: get_csr.stdout_lines | length > 0 + changed_when: certificate_approve.stdout + + - debug: # noqa name[missing] + msg: "{{ certificate_approve.stdout.split('\n') }}" + + + - name: Create test namespace + command: "{{ bin_dir }}/kubectl create namespace test" + changed_when: false + + - name: Wait for API token of test namespace + shell: "set -o pipefail && {{ bin_dir }}/kubectl describe serviceaccounts default --namespace test | grep Tokens | awk '{print $2}'" + args: + executable: /bin/bash + changed_when: false + register: default_token + until: default_token.stdout | length > 0 + retries: 5 + delay: 5 + + - name: Run 2 agnhost pods in test ns + shell: + cmd: | + set -o pipefail + cat <= groups['k8s_cluster'] | intersect(ansible_play_hosts) | length * 2 + retries: 3 + delay: 10 + failed_when: false + when: inventory_hostname == groups['kube_control_plane'][0] + + - name: Get netchecker pods + command: "{{ bin_dir }}/kubectl -n {{ netcheck_namespace }} describe pod -l app={{ item }}" + run_once: true + delegate_to: "{{ groups['kube_control_plane'][0] }}" + no_log: false + with_items: + - netchecker-agent + - netchecker-agent-hostnet + when: not nca_pod is success + + - debug: # noqa name[missing] + var: nca_pod.stdout_lines + when: inventory_hostname == groups['kube_control_plane'][0] + + - name: Get netchecker agents + uri: + url: "http://{{ ansible_default_ipv4.address }}:{{ netchecker_port }}/api/v1/agents/" + return_content: yes + run_once: true + delegate_to: "{{ groups['kube_control_plane'][0] }}" + register: agents + retries: 18 + delay: "{{ agent_report_interval }}" + until: agents.content | length > 0 and + agents.content[0] == '{' and + agents.content | from_json | length >= groups['k8s_cluster'] | intersect(ansible_play_hosts) | length * 2 + failed_when: false + no_log: false + + - name: Check netchecker status + uri: + url: "http://{{ ansible_default_ipv4.address }}:{{ netchecker_port }}/api/v1/connectivity_check" + status_code: 200 + return_content: yes + delegate_to: "{{ groups['kube_control_plane'][0] }}" + run_once: true + register: connectivity_check + retries: 3 + delay: "{{ agent_report_interval }}" + until: connectivity_check.content | length > 0 and + connectivity_check.content[0] == '{' + no_log: false + failed_when: false + when: + - agents.content != '{}' + + - debug: # noqa name[missing] + var: ncs_pod + run_once: true + + - name: Get kube-proxy logs + command: "{{ bin_dir }}/kubectl -n kube-system logs -l k8s-app=kube-proxy" + no_log: false + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not connectivity_check is success + + - name: Get logs from other apps + command: "{{ bin_dir }}/kubectl -n kube-system logs -l k8s-app={{ item }} --all-containers" + when: + - inventory_hostname == groups['kube_control_plane'][0] + - not connectivity_check is success + no_log: false + with_items: + - kube-router + - flannel + - canal-node + - calico-node + - cilium + + - name: Parse agents list + set_fact: + agents_check_result: "{{ agents.content | from_json }}" + delegate_to: "{{ groups['kube_control_plane'][0] }}" + run_once: true + when: + - agents is success + - agents.content is defined + - agents.content[0] == '{' + + - debug: # noqa name[missing] + var: agents_check_result + delegate_to: "{{ groups['kube_control_plane'][0] }}" + run_once: true + when: + - agents_check_result is defined + + - name: Parse connectivity check + set_fact: + connectivity_check_result: "{{ connectivity_check.content | from_json }}" + delegate_to: "{{ groups['kube_control_plane'][0] }}" + run_once: true + when: + - connectivity_check is success + - connectivity_check.content is defined + - connectivity_check.content[0] == '{' + + - debug: # noqa name[missing] + var: connectivity_check_result + delegate_to: "{{ groups['kube_control_plane'][0] }}" + run_once: true + when: + - connectivity_check_result is defined + + - name: Check connectivity with all netchecker agents + assert: + that: + - agents_check_result is defined + - connectivity_check_result is defined + - agents_check_result.keys() | length > 0 + - not connectivity_check_result.Absent + - not connectivity_check_result.Outdated + msg: "Connectivity check to netchecker agents failed" + delegate_to: "{{ groups['kube_control_plane'][0] }}" + run_once: true + + - name: Create macvlan network conf + # We cannot use only shell: below because Ansible will render the text + # with leading spaces, which means the shell will never find the string + # EOF at the beginning of a line. We can avoid Ansible's unhelpful + # heuristics by using the cmd parameter like this: + shell: + cmd: | + set -o pipefail + cat <