Code Review
- Download Kubernetes Learning Kit
- Unzip on your
HashiCorp
folder. - Use
~\HashiCorp\_Lecture_k8s_learning.kit-main\ch1\1.5\k8s-min-5GiB-wo-add-nodes
and~\HashiCorp\_Lecture_k8s_learning.kit-main\ch1\1.6
to see all settings files.
Vagrantfile
data:image/s3,"s3://crabby-images/31980/3198030f52446f143e2d90d96a0bec8d06b56080" alt=""
# -*- mode: ruby -*-
# vi: set ft=ruby :
- Vagrantfile is made by Ruby.
## configuration variables ##
# max number of worker nodes
N = 3
- Worker Nodes are three.
# each of components to install
k8s_V = '1.20.2' # Kubernetes
docker_V = '19.03.14-3.el7' # Docker
ctrd_V = '1.3.9-3.1.el7' # Containerd
## /configuration variables ##
- Versions for Kubernetes, Docker and ContainerD
Master Node
data:image/s3,"s3://crabby-images/fe830/fe830d7d90ab6ab2c1a09b889fa0080e20b261aa" alt=""
Vagrant.configure("2") do |config|
- Current Vagrantfile Api Version is 2.
do-end
statement will run line 19~35.
config.vm.define "WO-m-k8s-#{k8s_V[0..3]}" do |cfg|
k8s_V
is a variable for kubernetes version and0..3
means the index to read.- For example,
k8s_V
is 1.20.0 andk8s_V[0..3]
will get1.20
.
cfg.vm.box = "sysnet4admin/CentOS-k8s"
- VM image is CentOS private image.
cfg.vm.provider "virtualbox" do |vb|
vb.name = "WO-m-k8s-#{k8s_V[0..3]}(github_SysNet4Admin)"
vb.cpus = 2
vb.memory = 1746
vb.customize ["modifyvm", :id, "--groups", "/WO-k8s-SgMST-#{k8s_V[0..3]}(github_SysNet4Admin)"]
end
- Our virtual machine software is virtual box.
- Master Node has 2 cpus.
- Master Node has 1746 MB.
- We made a group to set on/off easy.
- If you want to see your customized groups, enter
kubectl get cm kubelet-config-1.22 -n kube-system -o yaml | grep cgroup
.
cfg.vm.host_name = "m-k8s"
cfg.vm.network "private_network", ip: "192.168.1.10"
cfg.vm.network "forwarded_port", guest: 22, host: 60010, auto_correct: true, id: "ssh"
cfg.vm.synced_folder "../data", "/vagrant", disabled: true
- Host name is
m-k8s
. private_network
makes connection network with your pc.forwarded_port
set local host to60010
.- When there is something wrong in our port, it will fix automatically.
- Port ID is
ssh
. synced_folder
will sync your pc folders with virtual machine folders, ifdisabled
is false.
cfg.vm.provision "shell", path: "k8s_env_build.sh", args: N
cfg.vm.provision "shell", path: "k8s_pkg_cfg.sh", args: [ k8s_V, docker_V, ctrd_V ]
#cfg.vm.provision "shell", path: "master_node.sh"
cfg.vm.provision "shell", path: "k_cfg_n_git_clone.sh" # add kubectl config & git clone source
- Run
xxx.sh
file in shell with arguments. - We seperated cluster part for practice.
Worker Node
data:image/s3,"s3://crabby-images/65f43/65f434339419e7e6c7cb61eafe5f61013f52c3e7" alt=""
(1..N).each do |i|
- Loop statement 1 to 3.
config.vm.define "WO-w#{i}-k8s-#{k8s_V[0..3]}" do |cfg|
cfg.vm.box = "sysnet4admin/CentOS-k8s"
cfg.vm.provider "virtualbox" do |vb|
vb.name = "WO-w#{i}-k8s-#{k8s_V[0..3]}(github_SysNet4Admin)"
vb.cpus = 1
vb.memory = 1024
vb.customize ["modifyvm", :id, "--groups", "/WO-k8s-SgMST-#{k8s_V[0..3]}(github_SysNet4Admin)"]
end
cfg.vm.host_name = "w#{i}-k8s"
cfg.vm.network "private_network", ip: "192.168.1.10#{i}"
cfg.vm.network "forwarded_port", guest: 22, host: "6010#{i}", auto_correct: true, id: "ssh"
cfg.vm.synced_folder "../data", "/vagrant", disabled: true
cfg.vm.provision "shell", path: "k8s_env_build.sh", args: N
cfg.vm.provision "shell", path: "k8s_pkg_cfg.sh", args: [ k8s_V, docker_V, ctrd_V ]
#cfg.vm.provision "shell", path: "work_nodes.sh"
- Worker Node has 1 cpu.
k8s_env_build.sh
- This file will set kubernetes environment.
data:image/s3,"s3://crabby-images/c612e/c612ef8c7bc9f3370e7580aa76797d29a4e76d22" alt=""
#!/usr/bin/env bash
- This script will use bash shell script.
# vim configuration
echo 'alias vi=vim' >> /etc/profile
vi
command is same withvim
# swapoff -a to disable swapping
swapoff -a
- Swap should be off to install kubernetes.
# sed to comment the swap partition in /etc/fstab
sed -i.bak -r 's/(.+ swap .+)/#\1/' /etc/fstab
- Swap is always off when you reboot the machine.
# kubernetes repo
gg_pkg="packages.cloud.google.com/yum/doc" # Due to shorten addr for key
cat <<EOF > /etc/yum.repos.d/kubernetes.repo
[kubernetes]
name=Kubernetes
baseurl=https://packages.cloud.google.com/yum/repos/kubernetes-el7-x86_64
enabled=1
gpgcheck=0
repo_gpgcheck=0
gpgkey=https://${gg_pkg}/yum-key.gpg https://${gg_pkg}/rpm-package-key.gpg
EOF
gpgcheck
is off.repo_gpgcheck
is off.- If you need security, you can set those to 1.
data:image/s3,"s3://crabby-images/57150/571504b4bc999feb001e0558c57b76f2931ef7ac" alt=""
# add docker-ce repo
yum install yum-utils -y
yum-config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo
- Use
yum
to download docker community repository.
# Set SELinux in permissive mode (effectively disabling it)
setenforce 0
sed -i 's/^SELINUX=enforcing$/SELINUX=permissive/' /etc/selinux/config
- SELinux if off.
- If you need security, you can set this to on.
# RHEL/CentOS 7 have reported traffic issues being routed incorrectly due to iptables bypassed
cat <<EOF > /etc/sysctl.d/k8s.conf
net.bridge.bridge-nf-call-ip6tables = 1
net.bridge.bridge-nf-call-iptables = 1
EOF
modprobe br_netfilter
br_netfilter
is bridge netfileter.br_netfilter
will connect your machines to one network.
# local small dns & vagrant cannot parse and delivery shell code.
echo "192.168.1.10 m-k8s" >> /etc/hosts
for (( i=1; i<=$1; i++ )); do echo "192.168.1.10$i w$i-k8s" >> /etc/hosts; done
- Deploy node name automatically, for example, m-k8s or w1-k8s.
$1
isk8s_v
.
# config DNS
cat <<EOF > /etc/resolv.conf
nameserver 1.1.1.1 #cloudflare DNS
nameserver 8.8.8.8 #Google DNS
EOF
- DNS settings.
k8s_env_build.sh
- This script is for installing kubernetes.
data:image/s3,"s3://crabby-images/1abf7/1abf751bd165c79b93581f1b47c15752a1c5a413" alt=""
# install util packages
yum install epel-release -y
yum install vim-enhanced -y
yum install git -y
epel-release
makes extended package for CentOs from Red Hat, examples, extended storage.vim-enhanced
will install vim.- You don’t need to intall
git
but for practice.
# install docker
yum install docker-ce-$2 docker-ce-cli-$2 containerd.io-$3 -y
$2
isdocker_V
and$3
isctrd_V
.
# install kubernetes
# both kubelet and kubectl will install by dependency
# but aim to latest version. so fixed version by manually
yum install kubelet-$1 kubectl-$1 kubeadm-$1 -y
$1
isk8s_V
.
# Ready to install for k8s
systemctl enable --now docker
systemctl enable --now kubelet
- Ready to system.
k_cfg_n_git_clone.sh
data:image/s3,"s3://crabby-images/bb282/bb2825ef594a002988889087f439b562f19aacb3" alt=""
# install bash-completion for kubectl
yum install bash-completion -y
# kubectl completion on bash-completion dir
kubectl completion bash >/etc/bash_completion.d/kubectl
bash-completion
allow us to use auto kubectl commands.
# alias kubectl to k
echo 'alias k=kubectl' >> ~/.bashrc
echo 'complete -F __start_kubectl k' >> ~/.bashrc
alias
makes short cut of commands.complete -F __start_kubectl k
allowk
to usebash-completion
# git clone k8s-code
git clone https://github.com/sysnet4admin/_Lecture_k8s_learning.kit.git
mv /home/vagrant/_Lecture_k8s_learning.kit $HOME
find $HOME/_Lecture_k8s_learning.kit -regex ".*\.\(sh\)" -exec chmod 700 {} \;
- Download practice codes from git.
WO_master_node.sh
data:image/s3,"s3://crabby-images/49823/49823d6990a93f3e7184b4b56ae020fe58539a4c" alt=""
# init kubernetes
kubeadm init --token 123456.1234567890123456 --token-ttl 0 \
--pod-network-cidr=172.16.0.0/16 --apiserver-advertise-address=192.168.1.10
- We need
token
to join Master Node and Worker Node. token-ttl
expires the token in 24 hours.pod-network-cidr
assigns pod’s network.apiserver-advertise-address
is fixed with Master Node IP address to avoid join problems.
# config for master node only
mkdir -p $HOME/.kube
cp -i /etc/kubernetes/admin.conf $HOME/.kube/config
chown $(id -u):$(id -g) $HOME/.kube/config
- To skip verification when we use kubectl.
# raw_address for gitcontent
raw_git="raw.githubusercontent.com/sysnet4admin/IaC/master/manifests"
# config for kubernetes's network
kubectl apply -f https://$raw_git/172.16_net_calico_v1.yaml
- Apply Calico for kubernetes network.
WO_work_nodes.sh
data:image/s3,"s3://crabby-images/80cc7/80cc734817400167e4deb95c4a58d07f04438d02" alt=""
# config for work_nodes only
kubeadm join --token 123456.1234567890123456 \
--discovery-token-unsafe-skip-ca-verification 192.168.1.10:6443
- Join with Master Node.
IDE
Deploy Kubernetes VM
- Open your
~HashiCorp\_Lecture_k8s_learning.kit-main\ch1\1.5\k8s-min-5GiB-wo-add-nodes
folder in command and usevagrant up
.
data:image/s3,"s3://crabby-images/49e2c/49e2c1787e5c350285078083b89231d8476f8569" alt=""
- Open your SuperPutty and click [File]-[Import Sessions]-[From File] to import
~HashiCorp\_Lecture_k8s_learning.kit-main\ch1\1.5\Sessions(k8s_learning).XML
.
Install Kubernetes with kubeadm
- Use
~/_Lecture_k8s_learning.kit/ch1/1.6/WO_master_node.sh
to install kubernetes in master node. - Open to worker node #1, #2 and #3 and use script of
~/_Lecture_k8s_learning.kit/ch1/1.6/WO_worker_node.sh
.
data:image/s3,"s3://crabby-images/993c0/993c038944f815b73f889258ac9db7ec476ae926" alt=""
- Open your
~HashiCorp\_Lecture_k8s_learning.kit-main\ch1\1.5\k8s-min-5GiB-wo-add-nodes>
folder and usevagrant destroy -f
to delete all VM. - We need to delete all VM to upgrade our VM.
IDE 2
- Update Kubernetes, Docker and ContainerD.
- Upgrade memory in master and worker nodes.
Deploy Kubernetes VM
- Open your
~HashiCorp\_Lecture_k8s_learning.kit-main\ch2\2.1\k8s-UpTo-10GiB
folder in command and usevagrant up
.
data:image/s3,"s3://crabby-images/458f3/458f355ddc757b960641e0913e30d8c3f7a2afd6" alt=""
Definitions
Object
- Container
-
Container has one software or system.
- Pod
- Pod has one or union of containers.
- Pod has a volume to save eternal data.
apiVersion: v1 # pod version
kind: Pod # Object type
metadata: # information of pod
labels:
run: po-nginx
name: po-nginx
spec: # spec of pod
containers: # information of container
- image: nginx
name: nginx
data:image/s3,"s3://crabby-images/854b5/854b5eb118dbe5c86ff98d52064250e3b3503fec" alt=""
data:image/s3,"s3://crabby-images/81672/8167257d2491b7daf699b196eb7d8ab3888d0443" alt=""
- Deployment
apiVersion: apps/v1 # deployment version
kind: Deployment # object type
metadata: # information of deployment
labels:
app: deploy-nginx
name: deploy-nginx
spec: # spec of deployment
replicas: 3 # replica set
selector: # choose templete
matchLabels:
app: po-nginx
template: # templete to make pod
metadata:
labels:
app: po-nginx
spec:
containers: # information of container
- name: nginx
image: nginx # container image
data:image/s3,"s3://crabby-images/03d89/03d89e13f6a92c88e017788a02abc908ab485a7f" alt=""
data:image/s3,"s3://crabby-images/e797c/e797c68a36b88c120b37c6ea0197c22bfb5014f9" alt=""
- ReplicaSet
- Deployment needs ReplicaSet to manage count of pods
apiVersion: apps/v1 # replicaset version
kind: ReplicaSet # object type
metadata: # information of replicaset
labels:
app: rs-nginx
name: rs-nginx
spec: # spec of replicaset
replicas: 3
selector: # choose templete
matchLabels:
app: po-nginx
template: # templete to make pod
metadata:
labels:
app: po-nginx
spec:
containers: # information of container
- image: nginx # container image
name: nginx
data:image/s3,"s3://crabby-images/66dfc/66dfcf05ec7003ff78919f2ec11c2d51967d902d" alt=""
- Honestly, the code is really similler with deployment, but we need ReplicaSet for rolling.
- For example, when you upgrade a pod, Deployment will create ReplicaSet, and ReplilcaSet will duplicate itself.
data:image/s3,"s3://crabby-images/36ba3/36ba31b571b8151995c0d23225fa3f95b861595b" alt=""
data:image/s3,"s3://crabby-images/d6839/d6839f18bc442ff8cf1b9e4301afaf54e715daaf" alt=""
- Job
- You can use job to decrese using memory.
apiVersion: batch/v1 # job version
kind: Job # object type
metadata: # information of job
name: job-curl-succ
spec: # spec of job
template: # templete to make job
spec:
containers: # information of container
- name: net-tools
image: sysnet4admin/net-tools # container image
command: ["curlchk", "nginx"]
restartPolicy: Never # restart option
restartPolicy
default value in other object isAlways
and it will restart the object forever.restartPolicy
should be in job and this value should beOnFailure
orNever
.
data:image/s3,"s3://crabby-images/8fc2c/8fc2cb2efdf368fa1d277d3af48464f22fc534ce" alt=""
data:image/s3,"s3://crabby-images/514f8/514f8d54ca0f4b8e41b94d428eb0058ffcd9f3b0" alt=""
apiVersion: batch/v1 # job version
kind: Job # object type
metadata: # information of job
name: job-completions
spec: # spec of job
completions: 3 # run sequentially 3 times
template: # templete to make job
spec:
containers:
- name: net-tools
image: sysnet4admin/net-tools # container image
command: ["curlchk", "nginx"]
restartPolicy: Never # restart option
- Use
completions
to run to sequentially.
data:image/s3,"s3://crabby-images/39f8f/39f8f83320a0b29bdf895c5e74221c0e3f385b4d" alt=""
apiVersion: batch/v1 # job version
kind: Job # object type
metadata: # information of job
name: job-parallelism
spec: # spec of job
parallelism: 3 # run parallely 3 times
template: # templete to make job
spec:
containers:
- name: net-tools
image: sysnet4admin/net-tools # container image
command: ["curlchk", "nginx"]
restartPolicy: Never # restart option
- Use
parallelism
to run to parallely.
data:image/s3,"s3://crabby-images/d3fa6/d3fa6ff82b2537f4ea1e59047213cb61f01b4be4" alt=""
apiVersion: batch/v1 # job version
kind: Job # object type
metadata: # information of job
name: job-activedeadlineseconds
spec: # spec of job
backoffLimit: 3
activeDeadlineSeconds: 30 # dead time after run command
template: # templete to make job
spec:
containers:
- name: net-tools
image: sysnet4admin/net-tools # container image
command: ["/bin/sh", "-c"]
args:
- sleep 60;
curlchk nginx;
restartPolicy: Never # restart option
- Use
activeDeadlineSeconds
to delete on specific time after your command
data:image/s3,"s3://crabby-images/57857/5785752bf0833e86157ad63dedc94f0de8733e98" alt=""
apiVersion: batch/v1 # job version
kind: Job # object type
metadata: # information of job
name: job-ttlsecondsafterfinished
spec: # spec of job
backoffLimit: 3
ttlSecondsAfterFinished: 30 # dead time after completed
template: # templete to make job
spec:
containers:
- name: net-tools
image: sysnet4admin/net-tools # container image
command: ["/bin/sh", "-c"]
args:
- sleep 60;
curlchk nginx;
restartPolicy: Never # restart option
- Use
ttlSecondsAfterFinished
to delete on specific time after completed
data:image/s3,"s3://crabby-images/78ba8/78ba895395d5cb70b4ed409b3dbd1edda09036d2" alt=""
- CronJob
- Use
CronJob
to run Job with schedule.
apiVersion: batch/v1 # cron job version
kind: CronJob # object type
metadata: # information of cron job
name: cj-1m-hist3-curl
spec: # spec of cron job
schedule: "*/1 * * * *" # cron rule
jobTemplate: # Template for job
spec: # same as before
template: # templete to make job
spec:
containers:
- name: net-tools
image: sysnet4admin/net-tools # container image
command: ["curlchk", "nginx"]
restartPolicy: Never # restart option
- cron rule :
*/#
repeats the job # periods, and just#
repeats the job at #.
data:image/s3,"s3://crabby-images/f6f21/f6f2176237949e6d659f3d8b3883fc76b6f6c2ee" alt=""
apiVersion: batch/v1 # cron job version
kind: CronJob # object type
metadata: # information of cron job
name: cj-1m-hist10-curl
spec: # spec of cron job
schedule: "*/1 * * * *" # cron rule
successfulJobsHistoryLimit: 10
jobTemplate: # Template for job
spec: # same as before
template: # templete to make job
spec:
containers:
- name: net-tools
image: sysnet4admin/net-tools # container image
command: ["curlchk", "nginx"]
restartPolicy: Never # restart option
successfulJobsHistoryLimit
hold the job until specific number. After the limited number, it will delete first job automatically. Default value is 3.
data:image/s3,"s3://crabby-images/f7f13/f7f135588f6f857f45da8a47932554a7f4910cb2" alt=""
-
Use
k get po | wc -l
to get total pods count. - DaemonSet
- DaemonSet makes one pod on each nodes.
apiVersion: apps/v1 # daemon set version
kind: DaemonSet # object type
metadata: # information of daemon set
labels:
app: ds-nginx
name: ds-nginx
spec: # spec of daemon set
selector:
matchLabels:
app: po-nginx
template: # Template for job
metadata:
labels:
app: po-nginx
spec: # same as before
containers:
- name: nginx
image: nginx # container image
- DaemonSet is quit simillar with deployment, but it has no replicas, because one pod can include only one DaemonSet.
data:image/s3,"s3://crabby-images/2da82/2da821f2bd8ceb4045bb21d4daf7bc447a2fdbb2" alt=""
- Use
vagrant up w4-k8s-1.22
to make fourth worker node.
data:image/s3,"s3://crabby-images/62815/62815db3f7b25a22fd496811c84cf560044bc566" alt=""
- Use
vagrant destroy -f w4-k8s-1.22
to destroy fourth worker node.
data:image/s3,"s3://crabby-images/0927d/0927d64210d604c7817ba0b753eee69835447f94" alt=""
-
When you add node, DamonSet will be created automatically from code.
- StatefulSet
- StatefulSet saves state of pod.
apiVersion: apps/v1 # StatefulSet version
kind: StatefulSet # object type
metadata: # information of StatefulSet
name: sts-chk-hn
spec: # spec of StatefulSet
replicas: 3
serviceName: sts-svc-domain #statefulset need it
selector:
matchLabels:
app: sts
template: # Template for StatefulSet
metadata:
labels:
app: sts
spec:
containers:
- name: chk-hn
image: sysnet4admin/chk-hn # container image
- You should use
serviceName
, because StatefulSet has specific name, not hash value.
data:image/s3,"s3://crabby-images/3767b/3767b61d06ae52cb7f61c59d976a7fe5c22ca6dd" alt=""
- Application
- Pod(s) containing container(s) and volume for specific work is(are) an application.
- For example, NGINX, MySQL, etc.
- Even when you add something on the application, that is also an application.
Commands
- get
-
Read object
- run, create, apply
- Create object
data:image/s3,"s3://crabby-images/66992/6699279a0827421b072c57933f770a78e08239f1" alt=""
- delete
-
Delet object
- exec
- Access to container in pod.
data:image/s3,"s3://crabby-images/c2588/c2588609e34b9ecda723a90955b709b058eaad0a" alt=""
- scale
- Add or sub count of pods.
data:image/s3,"s3://crabby-images/487fc/487fcd4ce59657c7ed2996dcb3fc7bf71092324c" alt=""
- edit
- Change deployed object.
data:image/s3,"s3://crabby-images/ba055/ba05535245d680d723c64826e5498fba332fc5ab" alt=""
- events
- Check events with namespace.
data:image/s3,"s3://crabby-images/630ee/630ee56e6e924c66c977774c0ba3b3bb0e69861d" alt=""
- describe
- Check status of object.
data:image/s3,"s3://crabby-images/b76ea/b76ea4500430014aa12f909030b1519eaf2e05ad" alt=""
- logs
- Check log.
- Log is worten when deploy is successed.
data:image/s3,"s3://crabby-images/39921/399218090f09d884e181f0ebd3fe255ef862f6d7" alt=""
yaml
- -o yaml
- Read yaml code.
data:image/s3,"s3://crabby-images/91ee5/91ee5b86c5a7e2eeaa966cf131ce08060730d905" alt=""
- –dry-run=client
- Run yaml code to read.
data:image/s3,"s3://crabby-images/1e8ac/1e8ac9e86ede413827a5521b7263abc54335e39d" alt=""
- command
- Use command in yaml file to run specific command.
data:image/s3,"s3://crabby-images/1f282/1f282bea95126dbaec119296e827f57a87c6be77" alt=""
- multiple commands
- Use
&&
to run multiple commands at once.
data:image/s3,"s3://crabby-images/fe2fb/fe2fb49c670080ce1d5082fcd60de8726320fd47" alt=""
- Use
;
to run multiple commans step by step.
data:image/s3,"s3://crabby-images/c6f94/c6f948716f018294327327f9a82db56ee348e4b9" alt=""
- Use
|
to separate command lines.
data:image/s3,"s3://crabby-images/dc73f/dc73f8d533d2555e9d5f41de0483dd1dba29d05a" alt=""
- Use
arg
to separate config and commands.
Expose Deployed Application
- We don’t use HostPort and HostNetwork, because we should know where the pods is running.
Port-forward
- We have host port and guest port. When we use host port, then this host port will be changed to guest port.
- For example, our host port is 60010, and when we connect to 60010, then our master node will change 60010 to 22 and connect to 22.
apiVersion: v1
kind: Pod
metadata:
name: fwd-chk-hn
spec:
containers:
- name: chk-hn
image: sysnet4admin/chk-hn
data:image/s3,"s3://crabby-images/bf3b2/bf3b2cd9c93d9ee5dd030b9c08b6019c7007afe1" alt=""
k port-forward fwd-chk-hn 80:80
means, we will open 80 with specific address and it will be changed to 80.k port-forward --address 0.0.0.0 fwd-chk-hn 80:80
means, we will open 80 with all address and it will be changed to 80.
HostPort
- Outside users should know, which node they will connect.
- For example, 8080 is second worker node host port, and it will change 8080 to 80.
apiVersion: v1
kind: Pod
metadata:
name: hp-chk-hn
spec:
containers:
- name: chk-hn
image: sysnet4admin/chk-hn
ports:
- containerPort: 80
hostPort: 8080
data:image/s3,"s3://crabby-images/53bc2/53bc2408513ccecc05b98368b51248452a54dab1" alt=""
HostNetwork
- Outside users should know, which node they will connect. And they connect directly that port.
apiVersion: v1
kind: Pod
metadata:
name: hnet-chk-hn
spec:
hostNetwork: true
containers:
- name: chk-hn
image: sysnet4admin/chk-hn
data:image/s3,"s3://crabby-images/af121/af12165476a2eb5f80b192d8fb453a0ce49eb0ea" alt=""
NodePort
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-nginx
labels:
app: deploy-nginx
spec:
replicas: 3
selector:
matchLabels:
app: deploy-nginx
template:
metadata:
labels:
app: deploy-nginx
spec:
containers:
- name: nginx
image: nginx
--- # separator for objects
apiVersion: v1
kind: Service
metadata:
name: np-nginx
spec:
selector:
app: deploy-nginx # deployment to expose
ports:
- name: http
port: 80 # service
targetPort: 80 # pod
nodePort: 30000 #option
type: NodePort # using NodePort
- User will connect 30000 node and node will connect to 80 service and service will connect to 80 pod.
data:image/s3,"s3://crabby-images/9d6eb/9d6eba51e3b4367d7e600d52b5bdd579a9f344f3" alt=""
LoadBalancer
- We will use matalib instead of NodePort.
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-nginx
labels:
app: deploy-nginx
spec:
replicas: 3
selector:
matchLabels:
app: deploy-nginx
template:
metadata:
labels:
app: deploy-nginx
spec:
containers:
- name: nginx
image: nginx
---
apiVersion: v1
kind: Service
metadata:
name: lb-nginx
spec:
selector:
app: deploy-nginx
ports:
- name: http
port: 80
targetPort: 80
type: LoadBalancer
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-chk-ip
labels:
app: deploy-chk-ip
spec:
replicas: 3
selector:
matchLabels:
app: deploy-chk-ip
template:
metadata:
labels:
app: deploy-chk-ip
spec:
containers:
- name: chk-ip
image: sysnet4admin/chk-ip
---
apiVersion: v1
kind: Service
metadata:
name: lb-chk-ip
spec:
selector:
app: deploy-chk-ip
ports:
- name: http
port: 80
targetPort: 80
type: LoadBalancer
data:image/s3,"s3://crabby-images/babe0/babe02ff668d4ef90a8c2795dc34afe80f9e9dcc" alt=""
ExternalName
- ExternalName has no Deployment because it uses external name to service.
- ExternalName is matched metadata’s name.
data:image/s3,"s3://crabby-images/8b85b/8b85ba16a9f6af823a24cd3b0757f1f4826506cb" alt=""
apiVersion: v1
kind: Service
metadata:
name: ex-url-1
namespace: default
spec:
type: ExternalName
externalName: sysnet4admin.github.io # External Domain Name
apiVersion: v1
kind: Service
metadata:
name: ex-url-2
namespace: default
spec:
type: ExternalName
externalName: k8s-edu.github.io # changable as you want
data:image/s3,"s3://crabby-images/73f03/73f030e0042350fd44f008a3db186d015687c7c8" alt=""
ClusterIP
- ClusterIP exposes Deployment or Pod.
data:image/s3,"s3://crabby-images/df307/df30758cacd7779b0709151fd89317d8547d9e98" alt=""
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-nginx
labels:
app: deploy-nginx
spec:
replicas: 3
selector:
matchLabels:
app: deploy-nginx
template:
metadata:
labels:
app: deploy-nginx
spec:
containers:
- name: nginx
image: nginx
---
apiVersion: v1
kind: Service
metadata:
name: cl-nginx
spec:
selector:
app: deploy-nginx
ports: # ClusterIP has port and target port to connect each other.
- name: http
port: 80
targetPort: 80
type: ClusterIP
Headless
- Headless exposes Deployment or Pod without IP.
data:image/s3,"s3://crabby-images/1349e/1349e9d14232da62f6a6f3bc59d3634fd38e7b19" alt=""
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-nginx
labels:
app: deploy-nginx
spec:
replicas: 3
selector:
matchLabels:
app: deploy-nginx
template:
metadata:
labels:
app: deploy-nginx
spec:
containers:
- name: nginx
image: nginx
---
apiVersion: v1
kind: Service
metadata:
name: hdl-nginx
spec:
selector:
app: deploy-nginx
ports:
- name: http
port: 80
targetPort: 80
clusterIP: None # Headless has no type and no IP.
data:image/s3,"s3://crabby-images/a1c63/a1c63ffa0dc8821a14dee24ef09e803288631091" alt=""
- Headless can communicate with domain name, without IP and connect to StateFulset with domain name.
- StateFulset matches service Name to connect Headless.
- When you use StateFulset with LoadBalancer, each External IP calls can show different pods. Therefore, I recommend, using StateFulset with Headless.
apiVersion: apps/v1
kind: StatefulSet
metadata:
name: sts-chk-hn
spec:
replicas: 3
serviceName: sts-svc-domain #statefulset need it
selector:
matchLabels:
app: sts
template:
metadata:
labels:
app: sts
spec:
containers:
- name: chk-hn
image: sysnet4admin/chk-hn
---
apiVersion: v1
kind: Service
metadata:
name: sts-svc-domain
spec:
selector:
app: sts
ports:
- port: 80
clusterIP: None
data:image/s3,"s3://crabby-images/feb17/feb177c87bb7c7e75d4198e359bdfb77e92c04cd" alt=""
EndPoint
- When you create Deployment and LoadBalancer together, EndPoint is also created.
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-chk-ip
labels:
app: deploy-chk-ip
spec:
replicas: 3
selector:
matchLabels:
app: deploy-chk-ip
template:
metadata:
labels:
app: deploy-chk-ip
spec:
containers:
- name: chk-ip
image: sysnet4admin/chk-ip
---
apiVersion: v1
kind: Service
metadata:
name: lb-chk-ip
spec:
selector:
app: deploy-chk-ip
ports:
- name: http
port: 80
targetPort: 80
type: LoadBalancer
data:image/s3,"s3://crabby-images/be74e/be74ed071f58f0ca56a04003b565cea25cf1599b" alt=""
- You can create EndPoint independently.
- Create Service first as ClusterIP and create also EndPoint with service name and LoadBalancer IP. As a result, you can call EndPoint with service name and EndPoint is binded with LoadBalancer IP, like double binding.
apiVersion: v1
kind: Service
metadata:
name: external-data
spec:
ports:
- name: http
port: 80
targetPort: 80
---
apiVersion: v1
kind: Endpoints
metadata:
name: external-data # match name with service.
subsets:
- addresses:
- ip: 192.168.1.11 # LoadBalancer IP
ports:
- name: http
port: 80
data:image/s3,"s3://crabby-images/2a058/2a058edac00cefa8dfcf07557995224f761d83ec" alt=""
Ingress
- Ingress cannot exist without service.
- Ingress has routing information and service routs the app.
data:image/s3,"s3://crabby-images/1cdd5/1cdd596a472024fb5b48f952a1a57f2a08ae5f49" alt=""
data:image/s3,"s3://crabby-images/c3496/c3496e6464936e97205160fb057aa9c2e730a54f" alt=""
-
Services
-
deploy-nginx
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-nginx
labels:
app: deploy-nginx
spec:
replicas: 3
selector:
matchLabels:
app: deploy-nginx
template:
metadata:
labels:
app: deploy-nginx
spec:
containers:
- name: nginx
image: nginx
---
apiVersion: v1
kind: Service
metadata:
name: ing-default
spec:
selector:
app: deploy-nginx
ports:
- name: http
port: 80
targetPort: 80
type: ClusterIP
- deploy-hn
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-hn
labels:
app: deploy-hn
spec:
replicas: 3
selector:
matchLabels:
app: deploy-hn
template:
metadata:
labels:
app: deploy-hn
spec:
containers:
- name: chk-hn
image: sysnet4admin/chk-hn
---
apiVersion: v1
kind: Service
metadata:
name: ing-hn
spec:
selector:
app: deploy-hn
ports:
- name: http
port: 80
targetPort: 80
type: ClusterIP
- deploy-ip
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-ip
labels:
app: deploy-ip
spec:
replicas: 3
selector:
matchLabels:
app: deploy-ip
template:
metadata:
labels:
app: deploy-ip
spec:
containers:
- name: chk-ip
image: sysnet4admin/chk-ip
---
apiVersion: v1
kind: Service
metadata:
name: ing-ip
spec:
selector:
app: deploy-ip
ports:
- name: http
port: 80
targetPort: 80
type: ClusterIP
- Ingress
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: nginx-ingress
annotations: # set default path
kubernetes.io/ingress.class: "nginx"
spec:
rules:
- http:
paths:
- path: /
pathType: Prefix
backend:
service:
name: ing-default
port:
number: 80
- path: /hn
pathType: Prefix
backend:
service:
name: ing-hn
port:
number: 80
- path: /ip
pathType: Prefix
backend:
service:
name: ing-ip
port:
number: 80
data:image/s3,"s3://crabby-images/a5a6f/a5a6f2cb7abc37eb137e638f1ae38d78487017b2" alt=""
- with NodePort
data:image/s3,"s3://crabby-images/4ddfc/4ddfc7be75ce12193272501ac196baa010816361" alt=""
data:image/s3,"s3://crabby-images/69d00/69d00b71b02a1f1c43cdd59e1d55f795d79bc39f" alt=""
- with LoadBalancer
data:image/s3,"s3://crabby-images/74915/749151a63f4530bfeb8ca81f791e0f772d6e54d9" alt=""
data:image/s3,"s3://crabby-images/bac00/bac00f2549412010bf7c2e5f83b3ea84afa7ffc7" alt=""
Label vs Annotation
- Label is for human and annotation is for system.
data:image/s3,"s3://crabby-images/ff448/ff448bb0412f85102a375573a5abafa991cc9f04" alt=""
Volume
emptyDir
- emptyDir is shared memory in pods.
apiVersion: v1
kind: Pod # make emptyDir as a pod
metadata:
name: pod-emptydir
labels:
app: nginx
spec:
containers:
- name: web-page # make container 1
image: nginx
volumeMounts:
- mountPath: /usr/share/nginx/html # container 1 first page
name: empty-directory
- name: html-builder # make container 2
image: alpine
volumeMounts:
- mountPath: /html-dir # container 2 first page is /html-dir/index.html
name: empty-directory
command: ["/bin/sh", "-c"]
args:
- echo "This page created on $(date +%Y-%m-%d)" > /html-dir/index.html;
sleep infinity;
volumes:
- name: empty-directory
emptyDir: {}
- In this example, we will create 2 containers.
- After creating each of first page, this first page gonna be empty-directory.
- This created empty-directory will be a volume and this volume connects 2 containers.
data:image/s3,"s3://crabby-images/22937/229371ca402da331620fce4d21fa794bb9c55b6c" alt=""
- We called container 2 with IP but actual running pod is container 1.
hostPath
- hostPath can use Node directories.
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-hostpath
labels:
app: deploy-hostpath
spec:
replicas: 3
selector:
matchLabels:
app: deploy-hostpath
template:
metadata:
labels:
app: deploy-hostpath
spec:
containers:
- name: host-mon
image: sysnet4admin/sleepy
volumeMounts:
- mountPath: /host-log
name: hostpath-directory
volumes:
- name: hostpath-directory
hostPath:
path: /var/log
- It connects
/var/log
and/host-log
with namehostpath-directory
. - When Deployment is released on Node, Node can take diffrent amounts of Deployments.
- So it’s difficult to make a hostPath on one Node.
data:image/s3,"s3://crabby-images/ec0c9/ec0c9da7d5941d69e30cb93afc70e4827c2cacbe" alt=""
apiVersion: apps/v1
kind: DaemonSet
metadata:
name: ds-hostpath
labels:
app: ds-hostpath
spec:
selector:
matchLabels:
app: ds-hostpath
template:
metadata:
labels:
app: ds-hostpath
spec:
containers:
- name: host-mon
image: sysnet4admin/sleepy
volumeMounts:
- mountPath: /host-log
name: hostpath-directory
volumes:
- name: hostpath-directory
hostPath:
path: /var/log
- DaemonSet is created on one Node. It means our hostPath is separated fairly on Nodes.
data:image/s3,"s3://crabby-images/3d8eb/3d8eb06877bb57b194f55a5dd4ba5681869c6ce4" alt=""
NFS
- Network File System
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-nfs
labels:
app: deploy-nfs
spec:
replicas: 3
selector:
matchLabels:
app: deploy-nfs
template:
metadata:
labels:
app: deploy-nfs
spec:
containers:
- name: chk-log
image: sysnet4admin/chk-log
volumeMounts:
- name: nfs-vol
mountPath: /audit
volumes:
- name: nfs-vol
nfs:
server: 192.168.1.10
path: /nfs_shared/nfs-vol
data:image/s3,"s3://crabby-images/4d416/4d41679df8e73c6a4f2b2fbcc7dd4d0b54fc8eb4" alt=""
PV & PVC
data:image/s3,"s3://crabby-images/73d3c/73d3ce1fbc9739df59d1714dce21d4ccc093a7fe" alt=""
- accessModes
- ReadWriteOnce(RWO) : Read and Write on only one node.
- ReadOnlyMany(ROX) : Read on several nodes.
- ReadWriteMany(RWX) : Read and Write on several nodes.
- Block Storage uses RWO and ROX and Object Storage uses RWX.
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-pvc
labels:
app: deploy-pvc
spec:
replicas: 3
selector:
matchLabels:
app: deploy-pvc
template:
metadata:
labels:
app: deploy-pvc
spec:
containers:
- name: chk-log
image: sysnet4admin/chk-log
volumeMounts:
- name: pvc-vol
mountPath: /audit
volumes:
- name: pvc-vol
persistentVolumeClaim:
claimName: pvc-nfs # created pvc name
PV
- Persistent Volume
apiVersion: v1
kind: PersistentVolume
metadata:
name: pv-nfs
spec:
capacity:
storage: 100Mi
accessModes:
- ReadWriteMany
persistentVolumeReclaimPolicy: Retain
nfs:
server: 192.168.1.10
path: /nfs_shared/pvc-vol
- persistentVolumeReclaimPolicy
- Retain : retain PV even you delete PVC
- Delete : delete PV when you delete PVC
PVC
- Persistent Volume Claim
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: pvc-nfs
spec:
accessModes:
- ReadWriteMany
resources:
requests:
storage: 10Mi
data:image/s3,"s3://crabby-images/6df3c/6df3c92eef0c5875caefd2af454537255a49d527" alt=""
StorageClass
- Persitent Volume Claim make StorageClass first, and then StorageClass make a Persistent Volume.
data:image/s3,"s3://crabby-images/de8c8/de8c81c188f15e7203cf69fa08a7815a6e03ca92" alt=""
data:image/s3,"s3://crabby-images/4e1a4/4e1a4f569f421f1a58087867fe0dab1ec0f6470d" alt=""
data:image/s3,"s3://crabby-images/aad08/aad08e6d2265af6535e5d33997e4d0551af47ed8" alt=""
- Provisioning
- Static : nfs, PV & PVC
- Dynamic : StorageClass
- In nfs mode, administrator should create pv everytime when it is requested.
- In PV & PVC mode, administrator should prepare always PV before user uses pvc.
- In StorageClass mode, administrator don’t need to prepare pvc before, because StorageClass will create PVC automatically everytime when it is requested.
data:image/s3,"s3://crabby-images/78070/780707b7830d859a9f0f7bcd98addb081aaaa805" alt=""
- provisioner
kind: Deployment
apiVersion: apps/v1
metadata:
name: nfs-client-provisioner
spec:
replicas: 1
selector:
matchLabels:
app: nfs-client-provisioner
strategy:
type: Recreate
template:
metadata:
labels:
app: nfs-client-provisioner
spec:
serviceAccountName: nfs-client-provisioner
containers:
- name: nfs-client-provisioner
image: k8s.gcr.io/sig-storage/nfs-subdir-external-provisioner:v4.0.2
volumeMounts:
- name: nfs-client-root
mountPath: /persistentvolumes
env:
- name: PROVISIONER_NAME
value: k8s-sigs.io/nfs-subdir-external-provisioner
- name: NFS_SERVER # NFS Server
value: 192.168.1.10
- name: NFS_PATH
value: /nfs_shared/dynamic-vol
volumes:
- name: nfs-client-root
nfs:
server: 192.168.1.10 # NFS Client Root
path: /nfs_shared/dynamic-vol
-
NFS server and NFS client Root should have same values.
-
StorageClass
apiVersion: storage.k8s.io/v1
kind: StorageClass
metadata:
name: managed-nfs-storage
# or choose another name, must match deployment's env PROVISIONER_NAME'
provisioner: k8s-sigs.io/nfs-subdir-external-provisioner
parameters:
# waits for nfs.io/storage-path annotation, if not specified will accept as empty string.
pathPattern: "${.PVC.namespace}/${.PVC.annotations.nfs.io/storage-path}"
onDelete: delete
- PVC
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: pvc-dynamic
spec:
accessModes:
- ReadWriteMany
resources:
requests:
storage: 10Gi
storageClassName: managed-nfs-storage # name of StorageClass
- Deployment
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-pvc
labels:
app: deploy-pvc
spec:
replicas: 3
selector:
matchLabels:
app: deploy-pvc
template:
metadata:
labels:
app: deploy-pvc
spec:
containers:
- name: chk-log
image: sysnet4admin/chk-log
volumeMounts:
- name: pvc-vol
mountPath: /audit
volumes:
- name: pvc-vol
persistentVolumeClaim:
claimName: pvc-dynamic # PVC name
data:image/s3,"s3://crabby-images/1806e/1806ef0e972663ff75e5d293def8be018b60d023" alt=""
data:image/s3,"s3://crabby-images/2ca07/2ca0788b46cb03cc50c99938ae9b4159a90e2985" alt=""
vol
- volumeClaimTemplates
- This is a volume type only for StateFullSet.
- StateFullSet has Status and independent Domain (with Headless). It means, StateFullSet accesses Pod independently.
- Therefore, each pod has unique value and status.
- When this pod claims volumeClaimTemplates, it creates independent PV.
- As a result, When you deploy with StateFullSet, your Pod has independent Domain, volumeClaimTemplates and PV.
data:image/s3,"s3://crabby-images/777a3/777a3dfe744b8f646604b456837a11485cb0ce30" alt=""
data:image/s3,"s3://crabby-images/7f88d/7f88dd7eb74cad2466becc07d0a424eca4e1b898" alt=""
data:image/s3,"s3://crabby-images/292c0/292c0f5599851ee9188aceec2528aa6f58e3fa6e" alt=""
apiVersion: apps/v1
kind: StatefulSet
metadata:
name: sts
spec:
replicas: 3
serviceName: sts-svc-domain #statefulset need it
selector:
matchLabels:
app: sts
template:
metadata:
labels:
app: sts
spec:
containers:
- name: chk-hn
image: sysnet4admin/chk-hn
volumeMounts:
- name: each-sts-backup # should be same with vol
mountPath: /backup_data
volumeClaimTemplates:
- metadata:
name: each-sts-backup
spec:
accessModes: [ "ReadWriteOnce" ] # cause it takes only one Pod
storageClassName: "managed-nfs-storage"
resources:
requests:
storage: 20Gi
data:image/s3,"s3://crabby-images/10b2d/10b2d2291f392f33268a9de853d7852d7b788e98" alt=""
data:image/s3,"s3://crabby-images/6448f/6448fec83d7dbde329f4ace0a247dfa2924484ec" alt=""
Node Contributions and Management
Cordon
data:image/s3,"s3://crabby-images/3f71d/3f71de7ca156986e0deed7c1c249d22e689edd9a" alt=""
- When you use cordon in specific node, that node is not affected from scheduling.
data:image/s3,"s3://crabby-images/c4913/c4913fe26fcda99106a0d3466a254d45abad16ee" alt=""
data:image/s3,"s3://crabby-images/437b7/437b789be085eace6574aaed792e100de7c1deec" alt=""
- w3-k8s is not updated, cause we set it with cordon.
Drain
data:image/s3,"s3://crabby-images/05495/0549583c06b0f190ffbf0da0d114ae1fc76063e4" alt=""
- Drain moves original pod to other nodes and set cordon on the node.
- Set drain on pod to maintanance or when the pod can occur some error.
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-drain
spec:
replicas: 3
selector:
matchLabels:
app: deploy-drain
template:
metadata:
labels:
app: deploy-drain
spec:
containers:
- name: nginx
image: nginx
data:image/s3,"s3://crabby-images/eb6c7/eb6c77efde1e6e3503ac94f3c371402e6045f8a9" alt=""
- At first, you will get this error to drain, because daemonset pod cannot be deleted.
- So you should use
--ignore-daemonsets --force
data:image/s3,"s3://crabby-images/2695b/2695b38060043ff448c8eff53b7029be7f75c1fb" alt=""
- And now you can see, we missed one pod
net
.
data:image/s3,"s3://crabby-images/2ca07/2ca0788b46cb03cc50c99938ae9b4159a90e2985" alt=""
nodeName
data:image/s3,"s3://crabby-images/3a114/3a114ac3b1738202005bcb3675296e61176ac826" alt=""
- Use nodeName to set where your pod should be deployed.
apiVersion: v1
kind: Pod
metadata:
name: nodename
spec:
containers:
- name: nginx
image: nginx
nodeName: w3-k8s # set nodeName where pod be deployed
data:image/s3,"s3://crabby-images/4e571/4e571214762b9e7c34b5955174283736c90744f6" alt=""
nodeLabel
data:image/s3,"s3://crabby-images/8b38d/8b38d8fba2b5ae5c726d88319b9d8c0ee80312f3" alt=""
- With nodeLabel and nodeSelector you can release several pods at once.
data:image/s3,"s3://crabby-images/914b0/914b0fbb674763c0c1f9a7e4678132884403d789" alt=""
- Use
k get node --show-labels
to see labels of nodes - Use
k label node [node] [label]
to add label on specific node
data:image/s3,"s3://crabby-images/8c0a1/8c0a1ea6b03b6ebcd04c11fb7240613ceaf4fd47" alt=""
- Use
k get node -l [label]
to search nodes with label =
symbol on label seperate key(left) value(right) and of course you can use only one when you search.
data:image/s3,"s3://crabby-images/e49af/e49af3949ff529c137508a990440f13765ddc570" alt=""
- Use
k label node [node] [label]-
to delete label on node
#!/usr/bin/env bash
kubectl label node w1-k8s gpupool=nvidia accelerator=tesla-a100
kubectl label node w2-k8s gpupool=nvidia accelerator=tesla-v100
kubectl label node w3-k8s diskint=nvme inmemory=redis
data:image/s3,"s3://crabby-images/9c2c3/9c2c36ee2abc3944e24a65abb18f715658d18096" alt=""
nodeSelector
- Use nodeSelector to set, in which node the pod should be deployed
data:image/s3,"s3://crabby-images/0458d/0458df0d7b817c752d8ba850caedaac82ed7f4f8" alt=""
apiVersion: v1
kind: Pod
metadata:
name: nodeselector-inmemory
spec:
containers:
- name: nginx
image: nginx
nodeSelector:
inmemory: redis # w3-k8s
data:image/s3,"s3://crabby-images/43765/437656eb60e5611550b9880be22d966f108850e3" alt=""
apiVersion: v1
kind: Pod
metadata:
name: nodeselector-gpupool
spec:
containers:
- name: nginx
image: nginx
nodeSelector:
gpupool: nvidia # w1-k8s, w2-k8s
data:image/s3,"s3://crabby-images/091c5/091c5011070083a934cea1fcf2749d1d0e8f2923" alt=""
nodeAffinity
- Use nodeAffinity to set more felexible confitions.
- There is two options to set: requiredDuringSchedulingIgnoredDuringExecution and preferredDuringSchedulingIgnoredDuringExecution
- Operators:
- In vs NotIn
- Exists vs DoesNotExist
- Gt vs Lt
apiVersion: v1
kind: Pod
metadata:
name: nodeaffinity
spec:
affinity:
nodeAffinity:
requiredDuringSchedulingIgnoredDuringExecution:
nodeSelectorTerms:
- matchExpressions: # inmemory-redis
- key: inmemory
operator: In
values:
- redis
containers:
- name: nginx
image: nginx
data:image/s3,"s3://crabby-images/6dc41/6dc412887458799c02922a3fa3c41fec7aeba533" alt=""
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: nodeaffinity-preferred
name: nodeaffinity-preferred
spec:
replicas: 3
selector:
matchLabels:
app: nodeaffinity-preferred
template:
metadata:
labels:
app: nodeaffinity-preferred
spec:
containers:
- image: nginx
name: nginx
affinity:
nodeAffinity: # It will search nodes who has gpupool-nvidia and prefer node who has accelerator-tesla-a100 to release
requiredDuringSchedulingIgnoredDuringExecution:
nodeSelectorTerms:
- matchExpressions: # gpupool-nvidia
- key: gpupool
operator: In
values:
- nvidia
preferredDuringSchedulingIgnoredDuringExecution:
- weight: 1 # higher means more affinity
preference:
matchExpressions: # accelerator-tesla-a100
- key: accelerator
operator: In
values:
- tesla-a100
data:image/s3,"s3://crabby-images/eaf79/eaf79b75b8a0530684b3ef02f8b0c886b07055b4" alt=""
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: anti-nodeaffinity
name: anti-nodeaffinity
spec:
replicas: 3
selector:
matchLabels:
app: anti-nodeaffinity
template:
metadata:
labels:
app: anti-nodeaffinity
spec:
containers:
- image: nginx
name: nginx
affinity:
nodeAffinity: # It will search nodes who has gpupool-nvidia and unprefer node who has accelerator-tesla-a100 to release
requiredDuringSchedulingIgnoredDuringExecution:
nodeSelectorTerms:
- matchExpressions: # gpupool-nvidia
- key: gpupool
operator: In
values:
- nvidia
preferredDuringSchedulingIgnoredDuringExecution:
- weight: 1
preference:
matchExpressions:
- key: accelerator # accelerator-tesla-a100 is unpreferred
operator: NotIn
values:
- tesla-a100
data:image/s3,"s3://crabby-images/ec8ef/ec8ef0fd2133196d30ccbc7d938562c67e01a42e" alt=""
Taints & Tolerations
data:image/s3,"s3://crabby-images/94291/942912f2187b060fee4206ae6eb85507e227b11f" alt=""
data:image/s3,"s3://crabby-images/171b6/171b61f497abe697ff1904d24c181d3951d28fd3" alt=""
- Effect
- NoSchedule : Only deploy with Telerations
- PreferNoSchedule : When there is no more nodes to deploy, ignore Taints setting
- NoExecute : Reschedule and delete pods which has no telerations
apiVersion: apps/v1
kind: DaemonSet
metadata:
name: daemonset-w-tolerations
labels:
app: daemonset-w-tolerations
spec:
selector:
matchLabels:
app: daemonset-w-tolerations
template:
metadata:
labels:
app: daemonset-w-tolerations
spec:
containers:
- name: nginx
image: nginx
tolerations:
- effect: NoSchedule
key: node-role.kubernetes.io/master
data:image/s3,"s3://crabby-images/c00e9/c00e9a5f536027370d8783b03923f7d23345072d" alt=""
- When key has kaster, then DaemonSet can be deployed in master also.
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: deploy-after-taints
name: deploy-after-taints
spec:
replicas: 6
selector:
matchLabels:
app: deploy-after-taints
template:
metadata:
labels:
app: deploy-after-taints
spec:
containers:
- image: nginx
name: nginx
data:image/s3,"s3://crabby-images/0b597/0b597938139ccec1e7fc14982f344328cc2cde2d" alt=""
data:image/s3,"s3://crabby-images/4a2cd/4a2cdcc659a4108884a3d7dcbd9107855dcfb37f" alt=""
- Pod cannot be deployed on w3-k8s, because w3-k8s has taint and pod has no teleration.
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: deploy-w-tolerations
name: deploy-w-tolerations
spec:
replicas: 6
selector:
matchLabels:
app: deploy-w-tolerations
template:
metadata:
labels:
app: deploy-w-tolerations
spec:
containers:
- image: nginx
name: nginx
tolerations: # it has telerations
- effect: NoSchedule
key: DB
value: customer-info
data:image/s3,"s3://crabby-images/ffd96/ffd9621824880fed81830eb441531a8fac884172" alt=""
data:image/s3,"s3://crabby-images/f7e7c/f7e7c5904436b032f929a3ea503c1a95fdfba3b0" alt=""
- Pod cab be deployed on w1-k8s, w2-k8s and w3-k8s, because pod has teleration.
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: deploy-w-tolerations-nodeaffinity
name: deploy-w-tolerations-nodeaffinity
spec:
replicas: 6
selector:
matchLabels:
app: deploy-w-tolerations-nodeaffinity
template:
metadata:
labels:
app: deploy-w-tolerations-nodeaffinity
spec:
containers:
- image: nginx
name: nginx
tolerations:
- effect: NoSchedule
key: DB
value: customer-info
affinity:
nodeAffinity:
requiredDuringSchedulingIgnoredDuringExecution:
nodeSelectorTerms:
- matchExpressions: # it prefer node who has inmemory-redis label
- key: inmemory
operator: In
values:
- redis
data:image/s3,"s3://crabby-images/39707/39707c89fae8908b0b170ad47e88e3ef1948da86" alt=""
data:image/s3,"s3://crabby-images/10686/106865d99c5c4ce397b3d699085ddfc0878c7f61" alt=""
#!/usr/bin/env bash
kubectl patch node w1-k8s -p '{"spec":{"taints":[]}}'
kubectl patch node w2-k8s -p '{"spec":{"taints":[]}}'
kubectl patch node w3-k8s -p '{"spec":{"taints":[]}}'
CODE=$(kubectl get node -o yaml | grep -i taints | wc -l) # Check status of taints
echo "successfully init taints in the k8s cluster"
echo "Result code is $CODE"
echo "if Result is not 1, please reload all of worker nodes"
- Above code is for deleting taints on nodes.
- Or you can jsut rerun nodes to delte taints.
data:image/s3,"s3://crabby-images/3f122/3f1224e57a24806e87a9983db1d7efed4244b22b" alt=""
Pod Composition and Management
Label
- Same with node label.
data:image/s3,"s3://crabby-images/f7010/f7010f391f77a2d9eccb97628eba007b3a88cfff" alt=""
data:image/s3,"s3://crabby-images/e3490/e3490297c1102b5faa08c0d9d1c831f77fa0a37b" alt=""
run=nginx
is created fromkubectl run
andapp=nginx
is created fromkubectl create
.
data:image/s3,"s3://crabby-images/fac1c/fac1c8d36e912d1c8c8d810fbba666bb9cc732ab" alt=""
- Use
k label pod [pod] [label]
to create custom label on pod.
Static Pod
data:image/s3,"s3://crabby-images/db96b/db96bd3f14f3b597ffecea98b61ee0a6b4e10168" alt=""
- Static Pod deploys etcd, controler manager and scheduler.
- Kubelet read yaml file and create api, etcd, customer manager and scheduler.
apiVersion: v1
kind: Pod
metadata:
name: static-pod
spec:
containers:
- name: nginx
image: nginx
data:image/s3,"s3://crabby-images/ab27d/ab27d85649987e6c92366376a0b8a33164a61da6" alt=""
data:image/s3,"s3://crabby-images/029ce/029ceb299724764773a361a77facdc51b029cd2c" alt=""
- Use
cp [Your Code Path] [Target Path]
to copy yaml code - Use
scp [Your Code Path] [Target Node]:[Target Path]
to copy yaml code on other node - Use
rm [Target Path]
to remove copied yaml code - You can only remove yaml code on accessed node. It means, to delete copied yaml code on other node, you should access other node.
restartPolicy
- Options
- Always : always restart
- Never : never restart
- OnFails : only restart when it failed
- Typo makes also restarting when the pod has OnFails option.
- Deployment only accept Always option, because Job is only once but deployment is continue.
apiVersion: v1
kind: Pod
metadata:
labels:
run: pod-always
name: pod-always
spec:
containers:
- image: sysnet4admin/net-tools
name: net-tools
command: ["/bin/sh", "-c"]
args:
- nslookup kubernetes
restartPolicy: Always
data:image/s3,"s3://crabby-images/0ba76/0ba76cc1abd77850e96b8fd7506607b4a536a826" alt=""
apiVersion: v1
kind: Pod
metadata:
labels:
run: pod-never
name: pod-never
spec:
containers:
- image: sysnet4admin/net-tools
name: net-tools
command: ["/bin/sh", "-c"]
args:
- nslookup kubernetes
restartPolicy: Never
data:image/s3,"s3://crabby-images/4097a/4097a9cee51ec297608197ca06a42a3726d7f553" alt=""
apiVersion: v1
kind: Pod
metadata:
labels:
run: pod-onfailure
name: pod-onfailure
spec:
containers:
- image: sysnet4admin/net-tools
name: net-tools
command: ["/bin/sh", "-c"]
args:
- nslookup kubernetes
restartPolicy: OnFailure
data:image/s3,"s3://crabby-images/a8fb7/a8fb74ab21ba7b40d12242d8bef60445b49e183f" alt=""
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: deploy-never-failure
name: deploy-never-failure
spec:
replicas: 1
selector:
matchLabels:
app: deploy-never-failure
template:
metadata:
labels:
app: deploy-never-failure
spec:
containers:
- name: nginx
image: nginx
restartPolicy: Never # Never cannot be used
data:image/s3,"s3://crabby-images/f6311/f6311f6d25355a95625f61713e906efe38201454" alt=""
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: deploy-onfailure-failure
name: deploy-onfailure-failure
spec:
replicas: 1
selector:
matchLabels:
app: deploy-onfailure-failure
template:
metadata:
labels:
app: deploy-onfailure-failure
spec:
containers:
- name: nginx
image: nginx
restartPolicy: OnFailure # Never cannot be used
data:image/s3,"s3://crabby-images/da74f/da74fa4ffb97b60a575e6227df3fb6dc6770a342" alt=""
Probe
- startupProbe
- Probe container status
- Kill container and execute with restartPolicy
- livenessProbe
- Probe container’s action
- Kill container and execute with restartPolicy
- readinessProbe
- Probe containers’s application whether can resolve requests
- Unpass traffic
livenessProbe
- Check Options
- exec : Execute container’s command
- httpGet : Get response from HTTP GET command
- tcpSocket : Check container’s address or port is alive
apiVersion: v1
kind: Pod
metadata:
labels:
run: liveness-exec
name: liveness-exec
spec:
containers:
- name: tardy-nginx
image: sysnet4admin/tardy-nginx
livenessProbe:
exec:
command:
- cat
- /tmp/healthy-on
initialDelaySeconds: 10
periodSeconds: 10 #it cannot start properly
data:image/s3,"s3://crabby-images/90b06/90b06803a8cf0dc72c9c4be2e79f1af79dca4cc3" alt=""
data:image/s3,"s3://crabby-images/332d6/332d6280a0b30e46028b6145d93fe81d0c7a765a" alt=""
watch "kubectl describe po liveness-exec | tail"
show changing continue- This cannot be worked, because initialDelaySeconds is 10 and preiodSeconds is also 10. It will repeat delay and initial infinity.
apiVersion: v1
kind: Pod
metadata:
labels:
run: liveness-httpget
name: liveness-httpget
spec:
containers:
- name: healthz-nginx
image: sysnet4admin/healthz-nginx
livenessProbe:
httpGet:
path: /healthz
port: 80
httpHeaders:
- name: purpose
value: health-check
initialDelaySeconds: 3
periodSeconds: 3
data:image/s3,"s3://crabby-images/6f4b1/6f4b135d390c6ef314087be58c9465585c2967c3" alt=""
data:image/s3,"s3://crabby-images/070b0/070b08faef5dbaca8e2b762a5441affa72f0e03c" alt=""
apiVersion: v1
kind: Pod
metadata:
labels:
run: liveness-tcpsocket
name: liveness-tcpsocket
spec:
containers:
- name: healthz-nginx
image: sysnet4admin/healthz-nginx
livenessProbe:
tcpSocket:
port: 80
initialDelaySeconds: 3
periodSeconds: 3
data:image/s3,"s3://crabby-images/069d1/069d19c78f8804ae853ee9ab3028765af8e17fa2" alt=""
readinessProbe
apiVersion: v1
kind: Pod
metadata:
labels:
run: readiness-exec
name: readiness-exec
spec:
containers:
- name: tardy-nginx
image: sysnet4admin/tardy-nginx
readinessProbe: // we will use a pod
exec:
command:
- cat
- /tmp/healthy-on
initialDelaySeconds: 10
periodSeconds: 5 // we will give short periodSecond.
---
apiVersion: v1
kind: Service // service is for removing endpoint.
metadata:
name: readiness-exec-lb
spec:
selector:
run: readiness-exec
ports:
- name: http
port: 80
targetPort: 80
type: LoadBalancer
- This application will not be killed because readinessProbe doesn’t rerun the application.
- readinessProbe just remove the endpoint.
data:image/s3,"s3://crabby-images/22432/224323ed6d2c7b9671aa05cd554c5f3586c0a572" alt=""
data:image/s3,"s3://crabby-images/bee76/bee76b83c0794f0a79adbf8f215cd8df26bde621" alt=""
- How readinessProbe remove entpoint
data:image/s3,"s3://crabby-images/bea52/bea5204acaf670866f74eaf1394d09fafe5380ab" alt=""
- How readinessProbe resume entpoint
data:image/s3,"s3://crabby-images/3da4c/3da4cb2bd691310916b03187efddeaf34f49ea89" alt=""
startupProbe
- startupProve is not used be alone because this is for bootup check.
apiVersion: v1
kind: Pod
metadata:
labels:
run: startup-w-others
name: startup-w-others
spec:
containers:
- name: tardy-nginx
image: sysnet4admin/tardy-nginx
startupProbe: // startupProbe
exec:
command:
- cat
- /tmp/healthy-on
initialDelaySeconds: 10
periodSeconds: 60
livenessProbe: // livenessProbe
exec:
command:
- cat
- /tmp/healthy-on
initialDelaySeconds: 10
periodSeconds: 10 // This is not matter because startupProbe has already made the image.
readinessProbe: // readinessProbe
exec:
command:
- cat
- /tmp/healthy-on
initialDelaySeconds: 5
periodSeconds: 5
data:image/s3,"s3://crabby-images/a7284/a7284509b69477ea45363e32d4ed5c9a2a81063c" alt=""
data:image/s3,"s3://crabby-images/2dba7/2dba7718ef60efc4783c735baf67813ee8580aa7" alt=""
data:image/s3,"s3://crabby-images/1b9f4/1b9f4358356b2915ded5639e0bd2f7477d7a70ba" alt=""
Init Container
- InitContainer make easier contratuctor for pod.
apiVersion: v1
kind: Pod
metadata:
name: pod-initcontainers
labels:
app: nginx
spec:
containers:
- name: web-page
image: nginx
volumeMounts:
- mountPath: /usr/share/nginx/html
name: empty-directory
initContainers:
- name: html-builder
image: alpine
volumeMounts:
- mountPath: /html-dir
name: empty-directory
command: ["/bin/sh", "-c"]
args:
- echo "This page created on $(date +%Y-%m-%d) by initContainers" > /html-dir/index.html;
volumes:
- name: empty-directory
emptyDir: {}
- pod initializing is from initContainer.
data:image/s3,"s3://crabby-images/17f3a/17f3a14b62bc0feff15017dc30662be425b09abe" alt=""
Multi Container
Sidecar
- First container make Web page and second container make server(e.g, NginX).
- This second container presents first conainer’s web page.
data:image/s3,"s3://crabby-images/6e5ce/6e5ce7b332948513be3791e437b84e772a058562" alt=""
Ambassador
- Second container is Proxy server and this second container takes over to present first container.
- It means, second container communicates with extern servers.
data:image/s3,"s3://crabby-images/4cc72/4cc7255d0510e0e65a48d70ea43deb9eb7cead55" alt=""
Adapter
- First container make data and second container translate this data.
- Second container expose this translated data to external.
data:image/s3,"s3://crabby-images/44dd7/44dd781487802eec5e5e7744919769438c4d54ce" alt=""
apiVersion: v1
kind: ConfigMap
metadata:
name: nginx-conf
data:
default.conf: |
server {
listen 80;
server_name nginx;
location / {
root /usr/share/nginx/html;
index index.html index.htm;
}
error_page 500 502 503 504 /50x.html;
location = /50x.html {
root /usr/share/nginx/html;
}
location /stub_status {
stub_status;
allow 127.0.0.1;
allow 192.168.1.0/24;
allow 172.16.0.0/16;
deny all; #deny all other hosts
}
}
apiVersion: v1
kind: Pod
metadata:
name: pod-adapter
labels:
app: nginx
spec:
containers:
- name: web-page
image: nginx
volumeMounts:
- mountPath: /etc/nginx/conf.d
name: nginx-conf
- name: adapter
image: nginx/nginx-prometheus-exporter:0.9.0
env:
- name: SCRAPE_URI
value: http://localhost/stub_status
ports:
- containerPort: 9113
volumes:
- name: nginx-conf
configMap:
name: nginx-conf
items:
- key: default.conf
path: default.conf
- In this case, first container is NginX(server) and second container is Prometheus(translator).
data:image/s3,"s3://crabby-images/e63b7/e63b7fef0efa0e62db045de475b867355da9fb0f" alt=""
data:image/s3,"s3://crabby-images/5eebe/5eebe7e987a0aa870754be599524d31c27167906" alt=""
Pod Affinity and Anti-Affinity
- You can use Pod affinity to group your pods.
- You can use Anti-Affinity to exclude your pods from groupping pods.
Affinity
apiVersion: v1
kind: Pod
metadata:
labels:
run: sleepy
affinity: leader
name: w1-affinity-leader
spec:
containers:
- image: sysnet4admin/sleepy
name: sleepy
nodeSelector:
kubernetes.io/hostname: w1-k8s
- Pod will be deployed on w1 always.
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: deploy-podaffinity
name: deploy-podaffinity
spec:
replicas: 4
selector:
matchLabels:
app: deploy-podaffinity
template:
metadata:
labels:
app: deploy-podaffinity
spec:
containers:
- image: nginx
name: nginx
affinity:
podAffinity:
requiredDuringSchedulingIgnoredDuringExecution:
- labelSelector:
matchExpressions:
- key: affinity
operator: In
values:
- leader
# If you want to change topologyKey,
# modify the admission controller, or disable.
topologyKey: kubernetes.io/hostname
data:image/s3,"s3://crabby-images/d521d/d521dfef8074da41baa82aa49e3ba4f8df84bdd7" alt=""
apiVersion: v1
kind: Pod
metadata:
labels:
run: sleepy
affinity: leader
name: w3-affinity-leader
spec:
containers:
- image: sysnet4admin/sleepy
name: sleepy
nodeSelector:
kubernetes.io/hostname: w3-k8s
- Newly created Pods will be deployed on w3 always.
data:image/s3,"s3://crabby-images/54d50/54d50393e9ff806841f1c7d9037cc48868951045" alt=""
Anti-Affinity
- Anti affinity will deploy pods, where has no affinity.
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: deploy-anti-podaffinity
name: deploy-anti-podaffinity
spec:
replicas: 4
selector:
matchLabels:
app: deploy-anti-podaffinity
template:
metadata:
labels:
app: deploy-anti-podaffinity
spec:
containers:
- image: nginx
name: nginx
affinity:
podAntiAffinity:
requiredDuringSchedulingIgnoredDuringExecution:
- labelSelector:
matchExpressions:
- key: affinity
operator: In
values:
- leader
# If you want to change topologyKey,
# modify the admission controller, or disable.
topologyKey: kubernetes.io/hostname
- In this case, this pods will be deployed on w2, because w1 and w3 has affinity already from previous commands.
data:image/s3,"s3://crabby-images/01b9a/01b9af3811df586644f4f3d11c8b9b31da1b6dd0" alt=""
TopologySpreadConstaints
- TopologySpreadConstaints can group pods with balance eventhough specific situations.
- At first, cluster read the count of nodes and set this all nodes as region.
- Then devides this nodes and set this devided noodes as zone.
data:image/s3,"s3://crabby-images/2fc61/2fc618cfa6274bbd501e147f1ad27d1aa6cca90e" alt=""
data:image/s3,"s3://crabby-images/32a23/32a239ec1b31922c110b0e7aafad112b10248002" alt=""
- Before we practice, we need one more worker node to make even.
data:image/s3,"s3://crabby-images/f73d6/f73d6e0f1da2880db07ee5513133c99ffaa60c9f" alt=""
#!/usr/bin/env bash
kubectl label node w1-k8s topology.kubernetes.io/region=ap-northeast-2 topology.kubernetes.io/zone=ap-northeast-2a
kubectl label node w2-k8s topology.kubernetes.io/region=ap-northeast-2 topology.kubernetes.io/zone=ap-northeast-2a
kubectl label node w3-k8s topology.kubernetes.io/region=ap-northeast-2 topology.kubernetes.io/zone=ap-northeast-2b
kubectl label node w4-k8s topology.kubernetes.io/region=ap-northeast-2 topology.kubernetes.io/zone=ap-northeast-2b
- This will create label on each node(e.gregion and zone).
data:image/s3,"s3://crabby-images/207c2/207c2c850375668ecebd806b714dc9fbc1361b00" alt=""
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: deploy-topologyspreadconstraints
name: deploy-topologyspreadconstraints
spec:
replicas: 4
selector:
matchLabels:
app: deploy-topologyspreadconstraints
template:
metadata:
labels:
app: deploy-topologyspreadconstraints
spec:
containers:
- image: nginx
name: nginx
topologySpreadConstraints:
- maxSkew: 1 // difference of each section should be not bigger then 1
topologyKey: topology.kubernetes.io/region // make same key
whenUnsatisfiable: DoNotSchedule // If this condition is false, it will not schedule any pods more.
labelSelector:
matchLabels:
app: deploy-topologyspreadconstraints // it will use this label, so pods are 4.
- maxSkew: 1
topologyKey: topology.kubernetes.io/zone // make same zone
whenUnsatisfiable: DoNotSchedule
labelSelector:
matchLabels:
app: deploy-topologyspreadconstraints
data:image/s3,"s3://crabby-images/69210/69210f842d7ae91db741f2f333b7eae9c07f071d" alt=""
- w2 has some pods before we try this command, so topology devide like, 2 pods on w1, 1 pod on w3 and 1 pod on w4.
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: deploy12-load-w3
name: deploy12-load-w3
spec:
replicas: 12
selector:
matchLabels:
app: deploy12-load-w3
template:
metadata:
labels:
app: deploy12-load-w3
spec:
containers:
- name: nginx
image: nginx
nodeName: w3-k8s
- This will make 12 pods on w3.
- And rerun topology, then it will devide like, 2 pods on w1 and 2 pods on w4.
data:image/s3,"s3://crabby-images/4ec71/4ec71718cb9661a85b05b091365fa0869e6b04ac" alt=""
data:image/s3,"s3://crabby-images/5a64f/5a64f5de76b22d35bfd2e36b1e5be87a2d039aff" alt=""
- Please remove w4 in virtual box!
data:image/s3,"s3://crabby-images/bb60e/bb60e64d212ab948d78321b15bbedf66cbb7b719" alt=""
Cluster Management
data:image/s3,"s3://crabby-images/35c0e/35c0e94cc30f127a7015238fa4526365b593b8e9" alt=""
data:image/s3,"s3://crabby-images/fbc6a/fbc6a50b90d7b8ad2efaee040841d1ecdc47dc93" alt=""
Access Control
RBAC(Role-Based Access Control)
data:image/s3,"s3://crabby-images/a93bc/a93bced7af2a1bb65f6032a4fa0ada47a2ec9c42" alt=""
- Node : set access permission from kubelet of scheduled node.
- ABAC : Attribute-based access control
- RBAC : set access permission from role.
- Webhook : Based on HTTP Post get Payload and control Authorization.
data:image/s3,"s3://crabby-images/63a3a/63a3a7adbe55c56170a4c45144f31d29a94f1c85" alt=""
-
Set role with behavior permission and set role group.
-
Context(Kubernetes Cluster)
data:image/s3,"s3://crabby-images/d1fd1/d1fd115a814718e6068da6a356dd3f31e912b494" alt=""
- dev1 is EKS(AWS).
- dev2 is AKS(Azure).
- dev3 is GKE(Google).
-
Context makes cluster and has access control data.
-
Practice
- Create namespace and account for dev1, dev 2 and cluster
data:image/s3,"s3://crabby-images/8d53e/8d53e64d8621645fd941944901239a2c9b73af01" alt=""
# dev1 namespace and account
apiVersion: v1
kind: Namespace
metadata:
name: dev1
---
apiVersion: v1
kind: ServiceAccount
metadata:
name: dev1-hoon
namespace: dev1
---
# dev2 namespace and account
apiVersion: v1
kind: Namespace
metadata:
name: dev2
---
apiVersion: v1
kind: ServiceAccount
metadata:
name: dev2-moon
namespace: dev2
# account for clusterrole
apiVersion: v1
kind: ServiceAccount
metadata:
name: sa-pod-admin
- Create Role and bind this role and account for dev1
- Role dev1 has get and list permission. So error is occured, when it try to create.
data:image/s3,"s3://crabby-images/60d16/60d16094505c05c0a1886f1d26a121dda4b46ed4" alt=""
data:image/s3,"s3://crabby-images/d4a58/d4a581cda0ab16ff018461604edbd1fe4fc9fcae" alt=""
data:image/s3,"s3://crabby-images/76c50/76c50a5776a5750cfa8d6dfa3bce5b8d8424ce22" alt=""
# dev1 Role
kind: Role
apiVersion: rbac.authorization.k8s.io/v1
metadata:
namespace: dev1
name: role-get-dev1
rules:
- apiGroups: ["*"]
resources: ["pods", "deployments"]
verbs: ["get", "list"]
# dev1 Role Binding
kind: RoleBinding
apiVersion: rbac.authorization.k8s.io/v1
metadata:
name: rolebinding-dev1
namespace: dev1
subjects:
- kind: ServiceAccount
name: dev1-hoon
apiGroup: ""
roleRef:
kind: Role
name: role-get-dev1
apiGroup: rbac.authorization.k8s.io
- Create Role and bind this role and account for dev2
- Role dev2 has get, list and create permission. So error is occured, when it try to delete.
data:image/s3,"s3://crabby-images/4e2fc/4e2fc2e49fcfdefd593ea4f23290c54a56e30997" alt=""
data:image/s3,"s3://crabby-images/9c02b/9c02be5057c9f9340ba6252ef805634878561b44" alt=""
# dev2 Role
kind: Role
apiVersion: rbac.authorization.k8s.io/v1
metadata:
namespace: dev2
name: role-gct-dev2
rules:
- apiGroups: ["*"]
resources: ["pods", "deployments"]
verbs: ["get", "list","create"]
# dev2 Role Binding
kind: RoleBinding
apiVersion: rbac.authorization.k8s.io/v1
metadata:
name: rolebinding-dev2
namespace: dev2
subjects:
- kind: ServiceAccount
name: dev2-moon
apiGroup: ""
roleRef:
kind: Role
name: role-gct-dev2
apiGroup: rbac.authorization.k8s.io
- Create Role and bind this role and account for cluster
- Role cluster has every thing on verb but it is accepted for pods, deployments, and deployment scale. So when it try to use service, it occures error.
data:image/s3,"s3://crabby-images/1d6a4/1d6a401cdd863e30820cf87846415fa742f82772" alt=""
data:image/s3,"s3://crabby-images/3a439/3a439a2d2c9bf8012b0c9c295c8ab3ad93766eaa" alt=""
# Cluster Role
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
name: pod-admin
rules:
- apiGroups: ["*"]
resources: ["pods","deployments","deployments/scale"]
verbs: ["*"]
# Cluster Role Binding
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRoleBinding
metadata:
name: clusterrolebinding-pod-admin
subjects:
- kind: ServiceAccount
name: sa-pod-admin
apiGroup: ""
# need namespace for CRB subjects
namespace: default
roleRef:
kind: ClusterRole
name: pod-admin
apiGroup: rbac.authorization.k8s.io
-
If you want to find with specific word, use “ grep”.
data:image/s3,"s3://crabby-images/964ae/964aea209123b9112976c2fb9b79d1550fee5a2a" alt=""
data:image/s3,"s3://crabby-images/dff86/dff86d77e9bc73f2105cf419c0e96574e03ccae4" alt=""
Resource Management
Resource Quota
data:image/s3,"s3://crabby-images/847ca/847ca326d02d792dc1385b533756be2f88f7df5c" alt=""
- Limitation for resource
apiVersion: v1
kind: ResourceQuota
metadata:
name: quota-dev1
namespace: dev1
spec:
hard:
pods: 10
managed-nfs-storage.storageclass.storage.k8s.io/persistentvolumeclaims: "2"
managed-nfs-storage.storageclass.storage.k8s.io/requests.storage: "2Gi"
#persistentvolumeclaims: "2"
#requests.storage: "2Gi"
-
So when we try to take attributes with more than limited value, it occures error.
-
Error with storage
data:image/s3,"s3://crabby-images/6cb35/6cb358fbd64df7961c97ed8e1fa32be1114de32f" alt=""
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: quota-3g-pvc-failure
namespace: dev1
spec:
accessModes:
- ReadWriteMany
resources:
requests:
storage: 3Gi
storageClassName: managed-nfs-storage
- Error with pvc
data:image/s3,"s3://crabby-images/5d2e0/5d2e0cf3751fe7ad21354cba98b6dde831c4d593" alt=""
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: quota-1g-pvc1
namespace: dev1
spec:
accessModes:
- ReadWriteMany
resources:
requests:
storage: 1Gi
storageClassName: managed-nfs-storage
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: quota-1g-pvc2
namespace: dev1
spec:
accessModes:
- ReadWriteMany
resources:
requests:
storage: 1Gi
storageClassName: managed-nfs-storage
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: quota-1g-pvc3
namespace: dev1
spec:
accessModes:
- ReadWriteMany
resources:
requests:
storage: 1Gi
storageClassName: managed-nfs-storage
- Error with pods
data:image/s3,"s3://crabby-images/a483f/a483f4215a1d9b0f0e406d5b14fa905f6c298d5e" alt=""
data:image/s3,"s3://crabby-images/9c4a4/9c4a47a989f8d67516cd413129f00039705d1b61" alt=""
- It will try to make 11 pods but error is occured when it try to make 11th pod.
apiVersion: apps/v1
kind: Deployment
metadata:
name: quota-pod11-failure
namespace: dev1
spec:
replicas: 11
selector:
matchLabels:
app: quota-pod11-failure
template:
metadata:
labels:
app: quota-pod11-failure
spec:
containers:
- name: nginx
image: nginx
data:image/s3,"s3://crabby-images/d42bd/d42bdc1a984589953ac06af2d3eb22e2c7dcc357" alt=""
data:image/s3,"s3://crabby-images/83f37/83f376de444b0834434ef92762f9722388433d3b" alt=""
data:image/s3,"s3://crabby-images/f24ee/f24ee85ae962cfbebe9f85280e84633ec6af8926" alt=""
- 176.jpg command is ‘k describe -n dev1 replicasets.apps quota-pod11-failure-7fc88499c7’
data:image/s3,"s3://crabby-images/346bc/346bc1d992c7a4c20304e84c9d638919cea3002e" alt=""
LimitRange
- LimitRange is effective object to process objective requests.
data:image/s3,"s3://crabby-images/20858/20858cfbf15d0a673b545c465db2195a4aca3388" alt=""
apiVersion: v1
kind: LimitRange
metadata:
name: limits-dev2
namespace: dev2
spec:
limits:
- type: PersistentVolumeClaim
max:
storage: 2Gi
min:
storage: 1Gi
- type: Container
default:
memory: 512Mi
defaultRequest:
memory: 256Mi
- Pod’s minimum memory size would be 256 Mi and maximum would be 512 Mi.
- PVC’s minimum memory size would be 1 Gi and maximum would be 2 Gi.
-
Like this, we can set LimitRange in namespace.
- G vs Gi
- For example, 5G means 5 Gigabytes while 5 Gibibytes.
- 5 G = 5000000 KB / 5000 MB
- 5 Gi = 5368709.12 KB / 5368.70 MB
data:image/s3,"s3://crabby-images/ae5f5/ae5f5a6bfe79d0e46ee841153175bc58dc311554" alt=""
- Now, dev2 has Limit Ranges with minimum 256 Mi and maximum 512 Mi.
- And when we request more resource on that namespace(our case is dev2), it occurs error like below.
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: limits-3g-pvc-failure
namespace: dev2
spec:
accessModes:
- ReadWriteMany
resources:
requests:
storage: 3Gi
storageClassName: managed-nfs-storage
data:image/s3,"s3://crabby-images/e7e78/e7e786a2b5fcf760e2fae7f6f011d4984e39a53d" alt=""
- But when we reauest appropriate resources, it will be created on namespace well.
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: limits-1g-pvc
namespace: dev2
spec:
accessModes:
- ReadWriteMany
resources:
requests:
storage: 1Gi
storageClassName: managed-nfs-storage
data:image/s3,"s3://crabby-images/648be/648be786332ed77b28da31a7cfe092235b093ef8" alt=""
- Extra experimence
apiVersion: apps/v1
kind: Deployment
metadata:
name: limits-defaultrequest
namespace: dev2
spec:
replicas: 6 # will be out of memory
selector:
matchLabels:
app: limits-defaultrequest
template:
metadata:
labels:
app: limits-defaultrequest
spec:
containers:
- name: chk-log
image: sysnet4admin/chk-log
volumeMounts:
- name: pvc-vol
mountPath: /audit
volumes:
- name: pvc-vol
persistentVolumeClaim:
claimName: limits-1g-pvc
nodeName: w3-k8s # only here for testing purpose
- Worker nodes have 1.5 Gi per each.
- We will test on w3-k8s to create 6 pods with minimum size 256 Mi.(6 x 256 Mi = 1.5 Gi)
- It should occur out of memory error because total needed memory is bigger than 1.5 Gi. (ex. pod memory)
data:image/s3,"s3://crabby-images/1b05e/1b05e770a516aa0cb893e5ac913708ec18c8023a" alt=""
- So please delete ASAP if you tested this code.
Network Policy
- Ingress Traffic : Traffic getting in to server through firewall
- Egress Traffic : Traffic getting out from server through firewall
data:image/s3,"s3://crabby-images/67722/67722ba15abd8f66c7a72f20523138006772d607" alt=""
Network Policy in Kubernetes
- Ingress : set direction of netrowk
data:image/s3,"s3://crabby-images/3fe4c/3fe4ce75a0ccb21486d58aa45c2dd9afd614c9a5" alt=""
- Before you run experiments, check you have net tools.
- If you don’t have net tools, run this
0-1-net-tools-ifn-default.yaml
and0-2-net-tools-ifn-dev[1-2].yaml
first.
apiVersion: v1
kind: Pod
metadata:
name: net
spec:
containers:
- image: sysnet4admin/net-tools-ifn
name: net
apiVersion: v1
kind: Pod
metadata:
name: net-dev1
namespace: dev1
spec:
containers:
- image: sysnet4admin/net-tools-ifn
name: net
---
apiVersion: v1
kind: Pod
metadata:
name: net-dev2
namespace: dev2
spec:
containers:
- image: sysnet4admin/net-tools-ifn
name: net
- Experiment 1 : Deny all
- if label’s role is sensitive, then deny all.
- This blocks every network transport in the pod.
- Because Ingress and Egress are just declared and there is no description for transporting.
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
role: sensitive
app: chk-info
name: deploy-deny-all
spec:
replicas: 3
selector:
matchLabels:
role: sensitive
app: chk-info
template:
metadata:
labels:
role: sensitive
app: chk-info
spec:
containers:
- image: sysnet4admin/chk-info
name: chk-info
---
apiVersion: v1
kind: Service
metadata:
labels:
app: chk-info
name: deploy-deny-all
spec:
ports:
- port: 80
protocol: TCP
targetPort: 80
selector:
app: chk-info
type: LoadBalancer
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
name: networkpolicy-deny-all
namespace: default
spec:
podSelector:
matchLabels:
role: sensitive
policyTypes:
- Ingress
- Egress
data:image/s3,"s3://crabby-images/ed618/ed618299b22efc70be6ebff9896dc1c68ac539a6" alt=""
-
It doesn’t work to get in the server, so please delete pods and policy after experiment.
-
Experiment 2 : Process with machted label
- if label’s role is internal, Ingress and Egress will be processed with mached label.
- In this case, Ingress will get in through chk-info app and Egress will get out though chk-info app.
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
name: networkpolicy-podselector
namespace: default
spec:
podSelector:
matchLabels:
role: internal
policyTypes:
- Ingress
- Egress
ingress:
- from:
- podSelector:
matchLabels:
app: chk-info
egress:
- to:
- podSelector:
matchLabels:
app: chk-info
data:image/s3,"s3://crabby-images/36c7a/36c7aeedcefb8ebf1f7cea4d248e0adf568a8866" alt=""
- Those pods can connect with only each others, it means transport occurs only inside of them.
- That is the reason, why we cannot connect the pod outside.
-
Please delete pods and policy before going to next expriment.
- Experiment 3-1 : Using IP block as criterion
podSelector:{}
means there is no criterion for label. So it will accept every labels.- In this case, Ingress will get in through ip 172.16.0.1 - 172.16.255.254 and Egress will get out through ip 172.16.0.1 - 172.16.127.254.
- So half of IPs cannot transport.
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
name: networkpolicy-ipblock
namespace: default
spec:
podSelector: {}
policyTypes:
- Ingress
- Egress
ingress:
- from:
- ipBlock:
# 172.16.0.1 - 172.16.255.254
cidr: 172.16.0.0/16
egress:
- to:
- ipBlock:
# 172.16.0.1 - 172.16.127.254
cidr: 172.16.0.0/17
data:image/s3,"s3://crabby-images/55934/559342f031981eab3e3690dc03a0853995ec7b62" alt=""
- I can use only
172.16.103.132
, others are over 127. - I executed net pod first with
k exec net -it -- /bin/bash
and then connected withping 172.16.103.132
. - Of course, I cannot connected with
ping 172.16.221.133
because of the policy. -
Please delete pods and policy for the next.
- Experiment 3-2 : Using IP block as criterion
- You can also except a specific IPs for transporting with
except
. - That excepted node should not be transported.
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
name: networkpolicy-ipblock-except
namespace: default
spec:
podSelector: {}
policyTypes:
- Ingress
- Egress
ingress:
- from:
- ipBlock:
cidr: 172.16.0.0/16
# change your CIDR to shut it down
#except:
# - 172.16.n.n/24
egress:
- to:
- ipBlock:
cidr: 172.16.0.0/16
# change your CIDR to shut it down
#except:
# - 172.16.n.n/24
data:image/s3,"s3://crabby-images/abe96/abe968352291a54c1fb7b33c86040ee9b3325f9c" alt=""
- In my case, I blocked
172.16.132.0/24
for Ingress and Egress. - You can also change above code with
vi
command. -
Please delete pods and policy for the next.
- Experiment 4 : Using namespace as criterion
- In this case, Ingress is only getting in through dev2.
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
name: networkpolicy-namespaceselector-dev2
namespace: dev2
spec:
podSelector: {}
policyTypes:
- Ingress
- Egress
ingress:
- from:
- namespaceSelector:
matchLabels:
kubernetes.io/metadata.name: dev2
egress:
- {}
data:image/s3,"s3://crabby-images/33311/333111231fca478ed9a8986826bb014370a7a7b8" alt=""
- So it doesn’t work, when you are not in dev2.
- But it works, when you are in dev2 because of the policy.
- Please delete pods and policy for the next.
data:image/s3,"s3://crabby-images/c6b07/c6b0788864f294c2913b7a6da1b500e1913ab78c" alt=""
Application Construction and Management
- Version upgrade
- Auto-Scale
- Deployment with Web UI
ConfigMap
- ConfigMap is for additional or changiable settings and messages.
apiVersion: v1
kind: ConfigMap
metadata:
name: sleepy-config
data:
STATUS: "SLEEP AGAIN"
NOTE: "TestBed Configuration"
# create a form with data type
data:image/s3,"s3://crabby-images/c1ee6/c1ee6c78bbbd5950c698bf409bdd062195e6ab0a" alt=""
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: deploy-configmapref
name: deploy-configmapref
spec:
replicas: 1
selector:
matchLabels:
app: deploy-configmapref
template:
metadata:
labels:
app: deploy-configmapref
spec:
containers:
- image: sysnet4admin/sleepy
name: sleepy
command: ["/bin/sh","-c"]
args:
- |
echo "sleepy $STATUS"
echo "NOTE: $NOTE"
sleep 3600
# $ means, we will use a specific variable here with same name
envFrom:
- configMapRef:
name: sleepy-config
# reference container 'sleepy-config' and it will be used as a environment variable
data:image/s3,"s3://crabby-images/3a85e/3a85ee6314cb7f172600e36fa113558352a5e8ee" alt=""
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: deploy-configmapkeyref
name: deploy-configmapkeyref
spec:
replicas: 1
selector:
matchLabels:
app: deploy-configmapkeyref
template:
metadata:
labels:
app: deploy-configmapkeyref
spec:
containers:
- image: sysnet4admin/sleepy
name: sleepy
command: ["/bin/sh","-c"]
args:
- |
echo "sleepy $APP_STATUS"
sleep 3600
env:
- name: APP_STATUS # mandantory field
valueFrom:
configMapKeyRef:
name: sleepy-config
key: STATUS
# reference key STATUS in 'sleepy-config' and this STATUS renamed to APP_STATUS
data:image/s3,"s3://crabby-images/6dcdd/6dcdd8faf6c97fcef13be310ccc25fd9d399d9de" alt=""
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: deploy-vol-configMap
name: deploy-vol-configmap
spec:
replicas: 1
selector:
matchLabels:
app: deploy-vol-configmap
template:
metadata:
labels:
app: deploy-vol-configmap
spec:
containers:
- image: sysnet4admin/sleepy
name: sleepy
command: ["/bin/sh","-c"]
args:
- |
sleep 3600
volumeMounts:
- name: appconfigvol
mountPath: /etc/sleepy.d
# mount ConfigMap as a volume
# read ConfigMap and mount it on the directory
# take values when we need
volumes:
- name: appconfigvol
configMap:
name: sleepy-config
data:image/s3,"s3://crabby-images/f6ccb/f6ccb3b0cd32516d545352ee8c8a7f0a9c232371" alt=""
- If ConfigMap is changed, then environment variables gonna be also changed.
apiVersion: v1
kind: ConfigMap
metadata:
name: sleepy-config
data:
STATUS: "SLEEP AGAIN" # changed
NOTE: "TestBed Configuration"
data:image/s3,"s3://crabby-images/1af95/1af95248aca7d55d9ebbda8daaa07c609587476e" alt=""
- Metallb can choose IP range by ConfigMap in load balance service.
apiVersion: v1
kind: ConfigMap
metadata:
namespace: metallb-system
name: config
data:
config: |
address-pools:
- name: metallb-ip-range
protocol: layer2
addresses:
- 192.168.1.11-192.168.1.19
data:image/s3,"s3://crabby-images/1df14/1df14e421ed11adefb99f1cc2f8a1bf8ffafd340" alt=""
Secret
- for security(ex. id or password)
- Secret Types
data:image/s3,"s3://crabby-images/105a2/105a23d1b632b9d7a1762748f2b9a56b6862dffc" alt=""
apiVersion: v1
kind: Secret
metadata:
name: mysql-cred
namespace: default
data: # base64
username: ZGItdXNlcg==
password: bGVlaHVobGVl
type: Opaque # default value
data:image/s3,"s3://crabby-images/d693d/d693d424a5d5acc3f2af2ca621eb558ce8245879" alt=""
- This endoding keys can be decoded by
echo {your key} | base64 --decode
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
run: deploy-secretkeyref
name: deploy-secretkeyref
spec:
replicas: 1
selector:
matchLabels:
app: deploy-secretkeyref
template:
metadata:
labels:
app: deploy-secretkeyref
spec:
containers:
- image: sysnet4admin/mysql-auth
name: mysql-auth
env:
# Need to init mysql
- name: MYSQL_ROOT_PASSWORD
value: rootpassword
# Custom auth
- name: MYSQL_USER_ID
valueFrom: # similar with ConfigMapRef
secretKeyRef:
name: mysql-cred # secret name
key: username # db-user
- name: MYSQL_USER_PASSWORD
valueFrom:
secretKeyRef:
name: mysql-cred
key: password # leehuhlee
ports:
- containerPort: 3306
volumeMounts:
- name: mysql-pvc
mountPath: /var/lib/mysql
volumes:
- name: mysql-pvc
persistentVolumeClaim:
claimName: mysql-pvc
---
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: mysql-pvc
spec:
accessModes:
- ReadWriteMany
resources:
requests:
storage: 20Gi
storageClassName: managed-nfs-storage
- Secret and ConfigMap can be changed with
kubectl edit
. - It means, when pod is dead and reveal, Secret and ConfigMap can be changed automatically.
- If Secret and ConfigMap should not or will not be changed, use
immutable
.
apiVersion: v1
kind: Secret
metadata:
name: mysql-cred
namespace: default
data:
username: ZGItdXNlcg==
password: bGVlaHVobGVl
type: Opaque
immutable: true
- After deploy secret, you cannot change this file again.
Roll Out
- Rolling Update
data:image/s3,"s3://crabby-images/0c537/0c53702d873f038bab2bd936836a872c65f80de0" alt=""
- Kubernetes updates 1 by 1 to continue services.
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-rollout
spec:
replicas: 3
selector:
matchLabels:
app: deploy-rollout
template:
metadata:
labels:
app: deploy-rollout
spec:
containers:
- name: nginx
image: nginx:1.20.1
data:image/s3,"s3://crabby-images/9ae4f/9ae4f191531376302de5ba2e341f91fff7bd9325" alt=""
data:image/s3,"s3://crabby-images/4b468/4b4689e0017073026aa92f7fdc593d83c29dab14" alt=""
data:image/s3,"s3://crabby-images/4ff33/4ff332321e46ae823a0f154a1b90823594dfb77c" alt=""
- update with unexisting version.
data:image/s3,"s3://crabby-images/c4569/c45694c7b020651527c780c2120389e8bd4bff78" alt=""
data:image/s3,"s3://crabby-images/cd198/cd19827fc63b953934c17561612de9088af51731" alt=""
data:image/s3,"s3://crabby-images/e26b1/e26b18667cd6f1eaed53efb94c8b96e3b2f5c9b8" alt=""
- update with specific version.
data:image/s3,"s3://crabby-images/4c8de/4c8def88a8a434a084deef7c43986bc878a559f3" alt=""
data:image/s3,"s3://crabby-images/acabd/acabd2948a8f3ffb7860ad51ced58ddb470beb33" alt=""
Kustomize
- Dynamic deplyoment
- Metallb can deploy several application at once.
apiVersion: policy/v1beta1
kind: PodSecurityPolicy
metadata:
labels:
app: metallb
name: controller
namespace: metallb-system
spec:
allowPrivilegeEscalation: false
allowedCapabilities: []
allowedHostPaths: []
defaultAddCapabilities: []
defaultAllowPrivilegeEscalation: false
fsGroup:
ranges:
- max: 65535
min: 1
rule: MustRunAs
hostIPC: false
hostNetwork: false
hostPID: false
privileged: false
readOnlyRootFilesystem: true
requiredDropCapabilities:
- ALL
runAsUser:
ranges:
- max: 65535
min: 1
rule: MustRunAs
seLinux:
rule: RunAsAny
supplementalGroups:
ranges:
- max: 65535
min: 1
rule: MustRunAs
volumes:
- configMap
- secret
- emptyDir
---
apiVersion: policy/v1beta1
kind: PodSecurityPolicy
metadata:
labels:
app: metallb
name: speaker
namespace: metallb-system
spec:
allowPrivilegeEscalation: false
allowedCapabilities:
- NET_RAW
allowedHostPaths: []
defaultAddCapabilities: []
defaultAllowPrivilegeEscalation: false
fsGroup:
rule: RunAsAny
hostIPC: false
hostNetwork: true
hostPID: false
hostPorts:
- max: 7472
min: 7472
- max: 7946
min: 7946
privileged: true
readOnlyRootFilesystem: true
requiredDropCapabilities:
- ALL
runAsUser:
rule: RunAsAny
seLinux:
rule: RunAsAny
supplementalGroups:
rule: RunAsAny
volumes:
- configMap
- secret
- emptyDir
---
apiVersion: v1
kind: ServiceAccount
metadata:
labels:
app: metallb
name: controller
namespace: metallb-system
---
apiVersion: v1
kind: ServiceAccount
metadata:
labels:
app: metallb
name: speaker
namespace: metallb-system
---
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
labels:
app: metallb
name: metallb-system:controller
rules:
- apiGroups:
- ''
resources:
- services
verbs:
- get
- list
- watch
- apiGroups:
- ''
resources:
- services/status
verbs:
- update
- apiGroups:
- ''
resources:
- events
verbs:
- create
- patch
- apiGroups:
- policy
resourceNames:
- controller
resources:
- podsecuritypolicies
verbs:
- use
---
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
labels:
app: metallb
name: metallb-system:speaker
rules:
- apiGroups:
- ''
resources:
- services
- endpoints
- nodes
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
---
apiVersion: rbac.authorization.k8s.io/v1
kind: Role
metadata:
labels:
app: metallb
name: config-watcher
namespace: metallb-system
rules:
- apiGroups:
- ''
resources:
- configmaps
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
---
apiVersion: rbac.authorization.k8s.io/v1
kind: Role
metadata:
labels:
app: metallb
name: controller
namespace: metallb-system
rules:
- apiGroups:
- ''
resources:
- secrets
verbs:
- create
- apiGroups:
- ''
resources:
- secrets
resourceNames:
- memberlist
verbs:
- list
- apiGroups:
- apps
resources:
- deployments
resourceNames:
- controller
verbs:
- get
---
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
---
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
---
apiVersion: rbac.authorization.k8s.io/v1
kind: RoleBinding
metadata:
labels:
app: metallb
name: config-watcher
namespace: metallb-system
roleRef:
apiGroup: rbac.authorization.k8s.io
kind: Role
name: config-watcher
subjects:
- kind: ServiceAccount
name: controller
- kind: ServiceAccount
name: speaker
---
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
---
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
---
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: '7472'
prometheus.io/scrape: 'true'
labels:
app: metallb
component: speaker
spec:
containers:
- args:
- --port=7472
- --config=config
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
# needed when another software is also using memberlist / port 7946
# when changing this default you also need to update the container ports definition
# and the PodSecurityPolicy hostPorts definition
#- name: METALLB_ML_BIND_PORT
# value: "7946"
- name: METALLB_ML_LABELS
value: "app=metallb,component=speaker"
- name: METALLB_ML_SECRET_KEY
valueFrom:
secretKeyRef:
name: memberlist
key: secretkey
image: quay.io/metallb/speaker:main
name: speaker
ports:
- containerPort: 7472
name: monitoring
- containerPort: 7946
name: memberlist-tcp
- containerPort: 7946
name: memberlist-udp
protocol: UDP
securityContext:
allowPrivilegeEscalation: false
capabilities:
add:
- NET_RAW
drop:
- ALL
readOnlyRootFilesystem: true
hostNetwork: true
nodeSelector:
kubernetes.io/os: linux
serviceAccountName: speaker
terminationGracePeriodSeconds: 2
tolerations:
- effect: NoSchedule
key: node-role.kubernetes.io/master
operator: Exists
---
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: '7472'
prometheus.io/scrape: 'true'
labels:
app: metallb
component: controller
spec:
containers:
- args:
- --port=7472
- --config=config
env:
- name: METALLB_ML_SECRET_NAME
value: memberlist
- name: METALLB_DEPLOYMENT
value: controller
image: quay.io/metallb/controller:main
name: controller
ports:
- containerPort: 7472
name: monitoring
securityContext:
allowPrivilegeEscalation: false
capabilities:
drop:
- all
readOnlyRootFilesystem: true
nodeSelector:
kubernetes.io/os: linux
securityContext:
runAsNonRoot: true
runAsUser: 65534
fsGroup: 65534
serviceAccountName: controller
terminationGracePeriodSeconds: 0
Kustomize process
data:image/s3,"s3://crabby-images/71095/710957fb50c62aa507885f33e19d66284762996c" alt=""
- We can change dynamically kustomization.yaml file with sources by
kustomize create
. - And then we can create upgrade build file with
kustomize build
. - Finally you can upgrade with
kubectl apply -f
.
data:image/s3,"s3://crabby-images/3fa81/3fa8158ef6759c4eec47412b71e25645616bd9aa" alt=""
- Delete MetalLB
data:image/s3,"s3://crabby-images/76e2b/76e2b681dfd85aeb6fc9d22c8aaf5e5da780cad5" alt=""
- Check MetalLB is deleted and install kustomize
data:image/s3,"s3://crabby-images/6bfb2/6bfb2c2abf7eeff91b6c9ed3c4b3edb5a61b9658" alt=""
- Create kustomize and edit version
data:image/s3,"s3://crabby-images/33c84/33c8465eff8b54dbfbb0a4dada62239c5fb5e34d" alt=""
- Deploy MetalLB
data:image/s3,"s3://crabby-images/768a7/768a7b3f099d33e6ea538493d9ad2fe7d0d31bfb" alt=""
- Chcek MetalLB is deployed
helm
- helm make dynamic deploy easier.
- helm > kustomize.io > kubectl
- User’s work with helm is just setting storage and intalling release
data:image/s3,"s3://crabby-images/e4ae4/e4ae4ccd9eb11e75b319592a2e76cc03ec7e63cd" alt=""
- Install helm
data:image/s3,"s3://crabby-images/b0a8f/b0a8f5a1d01c0a2ffd68fcf28b5ca118a531e7c6" alt=""
- Set storage
data:image/s3,"s3://crabby-images/cf48f/cf48f8a2ca971986cfa89d1c1feee06978103042" alt=""
- Delete oldest metalLB and NFS provisioner because we will use only helm
data:image/s3,"s3://crabby-images/eef5a/eef5a4b9b4741de4c549848da86a4a1157e5008b" alt=""
data:image/s3,"s3://crabby-images/50696/5069653e95b0f6002c677670d2e127b0435cbce7" alt=""
- Install metalLB with helm
- This is
metallb-installer-by-helm.sh
helm install metallb edu/metallb \
--create-namespace \
--namespace=metallb-system \
--set controller.image.tag=v0.10.2 \
--set speaker.image.tag=v0.10.2 \
-f ~/_Lecture_k8s_learning.kit/ch9/9.6/installer-by-helm/l2-config-by-helm.yaml
- helm will install metalLB with namespace
metallb-system
and image version0.10.2
. - helm will apply config with this.
configInline:
address-pools:
- name: default
protocol: layer2
addresses:
- 192.168.1.11-192.168.1.49
data:image/s3,"s3://crabby-images/78e3f/78e3f685d93bbf807ad49a2fb8fd4311ea4c7a50" alt=""
- Install metalLB with helm
- This is
nfs-provisioner-installer-by-helm.sh
helm install nfs-provisioner edu/nfs-subdir-external-provisioner \
--set nfs.server=192.168.1.10 \
--set nfs.path=/nfs_shared/dynamic-vol \
--set storageClass.name=managed-nfs-storage
- Uninstall metalLB with helm, if you need.
data:image/s3,"s3://crabby-images/7a5bf/7a5bfb9c78ecfe25dbeb1ccc5215dbf14aaf53cf" alt=""
Metrics-Server
- Mornitoring resources, like CPU and memory.
- Each kubelet in worker node sends measured value to Metrics-Server.
- And user can get measured value from Metrics-Server with API.
data:image/s3,"s3://crabby-images/889d3/889d39bc09ae9f0cb9172e4cb00c7fdf91f9d377" alt=""
- If current system has not kustomization,
kubectl apply -k .
will read and excute all referenced kustomization files. - You can see measured value by
k top
data:image/s3,"s3://crabby-images/d028c/d028cc8cf28d9a8f67467c864093dd27508c80b4" alt=""
data:image/s3,"s3://crabby-images/a96a2/a96a208a8153a07225bff0b4a21c9b5aced0cd2c" alt=""
data:image/s3,"s3://crabby-images/6a1e9/6a1e9adc8d031b5a9cf056ed633b1c441554b810" alt=""
HPA (Horizontal Pod Autoscaler)
- Sync with Metrics-Server
- Scale pods automatically
- Applications will be managed automatically depends on kubernetes resources’ status.
- HPA is for linear load increasing or predictable load.
data:image/s3,"s3://crabby-images/f6a3c/f6a3c9659efc79fd239a76b33c9696091eb80c02" alt=""
- API Server sends measured value about pods to HPA.
- HPA applies resources to appropriate deployment when they has to be needed.
apiVersion: apps/v1
kind: Deployment
metadata:
name: deploy-4-hpa
labels:
app: deploy-4-hpa
spec:
replicas: 1
selector:
matchLabels:
app: deploy-4-hpa
template:
metadata:
labels:
app: deploy-4-hpa
spec:
containers: # Set resources for containers
- name: chk-hn
image: sysnet4admin/chk-hn
resources:
requests: # min
cpu: "10m"
limits: # max
cpu: "20m"
---
apiVersion: v1
kind: Service
metadata:
name: lb-deploy-4-hpa
spec:
selector:
app: deploy-4-hpa
ports:
- name: http
port: 80
targetPort: 80
type: LoadBalancer
- Active HPA with below code or command
k autoscale deployment deploy-4-hpa --min=1 --max=10 --cpu-percent=50
- You can create below code by
k autoscale deployment deploy-4-hpa --min=1 --max=10 --cpu-percent=50 --dry-run=client -o yaml
apiVersion: autoscaling/v1
kind: HorizontalPodAutoscaler
metadata:
name: deploy-4-hpa
spec:
maxReplicas: 10
minReplicas: 1
scaleTargetRef:
apiVersion: apps/v1
kind: Deployment
name: deploy-4-hpa
targetCPUUtilizationPercentage: 50
data:image/s3,"s3://crabby-images/e452a/e452a7ddcfa5d034c73a2457e10746660b87e84a" alt=""
- Set load
while true
do
COUNTER=$((COUNTER + 1))
echo -ne "$COUNTER - " ; curl $1
done
Excersice
- Apply deployment and autoscale.
data:image/s3,"s3://crabby-images/c32d9/c32d90e231cfbe82cfe2221ebdc5fde045124ec2" alt=""
data:image/s3,"s3://crabby-images/7a8ef/7a8ef8e429cca83459f60e3cba42ab44899e7116" alt=""
- Monitor pods by
watch kubectl top pods --use-protocol-buffers
- Monitor HPA by
watch kubectl get hpa
data:image/s3,"s3://crabby-images/53b99/53b99be3771cd82180d0758a381760f75af0576a" alt=""
- Set load
data:image/s3,"s3://crabby-images/b5a8c/b5a8c03b2236687e2fceeaf9c2114a0ba78d8f28" alt=""
data:image/s3,"s3://crabby-images/b13f1/b13f1fb4634677d570e3604e4a23e4fd200f5ea3" alt=""
data:image/s3,"s3://crabby-images/cae97/cae9753cf17cfcd5219310a41f735cabdb9be108" alt=""
- Stop load
data:image/s3,"s3://crabby-images/66505/6650577909890feca4722962df2abd588ae7aa62" alt=""
data:image/s3,"s3://crabby-images/51d5d/51d5d701468527cd726e362e6c7e82a320bc1429" alt=""
kube-dashboard
- You can do all commands on Web UI.
Excercise
- Check kubernetes-dashboard service IP.
data:image/s3,"s3://crabby-images/66009/660092693d80494bc192856dc7451b94f9ea4bdd" alt=""
- Open Web UI with kubernetes-dashboard service IP.
- Click skip for authorization.
data:image/s3,"s3://crabby-images/279d3/279d34ab033972df0e481fce8490f41a4a7b0a52" alt=""
data:image/s3,"s3://crabby-images/5ff83/5ff83b0a2fca066009a8c0b3a4335f52ca0d841d" alt=""
- Create a pod with left-top
+
button. - Click deploy button.
data:image/s3,"s3://crabby-images/fa762/fa7621779f5dbcfd08139ac7a552d7678b4e1a35" alt=""
data:image/s3,"s3://crabby-images/76f8b/76f8b5fb5817ac37d2ca3ecfb1cf8fe68d3778c8" alt=""
data:image/s3,"s3://crabby-images/6b56a/6b56abe2c90d8cfaba2b790ab4c051df25894357" alt=""
- Click Pods in bread.
- Click three dots button to see information of pod-web.
data:image/s3,"s3://crabby-images/85310/8531045500da42597f1b0aab6380d25e8e889023" alt=""
data:image/s3,"s3://crabby-images/1f486/1f486c93c914b3e1671bb4f2e7fc6d66c409525c" alt=""
data:image/s3,"s3://crabby-images/174c8/174c886ad54cf342520ae9ab763da507c6c55fc3" alt=""
- Click deployment
- Delete pod-web deployment with three dots button.
data:image/s3,"s3://crabby-images/7ccfe/7ccfe2fde57b6c37da83049c2e265d39f6b617d0" alt=""
data:image/s3,"s3://crabby-images/c6e92/c6e927eed16574e151d5a856484a23812eeffee0" alt=""
data:image/s3,"s3://crabby-images/8f4b8/8f4b800fbdc5b69c82a2bd65b4840ee9c17f8d83" alt=""