From 44863a46fae83354304a65003c308f3568037ec9 Mon Sep 17 00:00:00 2001 From: NotHarshhaa Date: Sun, 3 Sep 2023 17:40:16 +0530 Subject: [PATCH] IntoTheDevOps: Add End-to-End Kubernetes content Signed-off-by: NotHarshhaa --- topics/kubernetes/CKA.md | 868 +++++ topics/kubernetes/README.md | 3157 +++++++++++++++++ .../kustomize_common_labels/exercise.md | 17 + .../kustomize_common_labels/solution.md | 31 + .../someApp/deployment.yml | 21 + .../someApp/service.yml | 11 + .../labels_and_selectors/exercise.md | 11 + .../labels_and_selectors/solution.md | 13 + .../exercises/node_selectors/exercise.md | 12 + .../exercises/node_selectors/solution.md | 29 + .../exercises/taints_101/exercise.md | 13 + .../exercises/taints_101/solution.md | 30 + .../images/cluster_architecture_exercise.png | Bin 0 -> 46202 bytes .../images/cluster_architecture_solution.png | Bin 0 -> 44343 bytes topics/kubernetes/images/service_exercise.png | Bin 0 -> 27380 bytes topics/kubernetes/images/service_solution.png | Bin 0 -> 25341 bytes topics/kubernetes/killing_containers.md | 12 + topics/kubernetes/pods_01.md | 11 + topics/kubernetes/replicaset_01.md | 14 + topics/kubernetes/replicaset_02.md | 12 + topics/kubernetes/replicaset_03.md | 14 + topics/kubernetes/services_01.md | 11 + .../solutions/killing_containers.md | 12 + .../kubernetes/solutions/pods_01_solution.md | 6 + .../solutions/replicaset_01_solution.md | 62 + .../solutions/replicaset_02_solution.md | 62 + .../solutions/replicaset_03_solution.md | 61 + .../solutions/services_01_solution.md | 19 + 28 files changed, 4509 insertions(+) create mode 100644 topics/kubernetes/CKA.md create mode 100644 topics/kubernetes/README.md create mode 100644 topics/kubernetes/exercises/kustomize_common_labels/exercise.md create mode 100644 topics/kubernetes/exercises/kustomize_common_labels/solution.md create mode 100644 topics/kubernetes/exercises/kustomize_common_labels/someApp/deployment.yml create mode 100644 topics/kubernetes/exercises/kustomize_common_labels/someApp/service.yml create mode 100644 topics/kubernetes/exercises/labels_and_selectors/exercise.md create mode 100644 topics/kubernetes/exercises/labels_and_selectors/solution.md create mode 100644 topics/kubernetes/exercises/node_selectors/exercise.md create mode 100644 topics/kubernetes/exercises/node_selectors/solution.md create mode 100644 topics/kubernetes/exercises/taints_101/exercise.md create mode 100644 topics/kubernetes/exercises/taints_101/solution.md create mode 100644 topics/kubernetes/images/cluster_architecture_exercise.png create mode 100644 topics/kubernetes/images/cluster_architecture_solution.png create mode 100644 topics/kubernetes/images/service_exercise.png create mode 100644 topics/kubernetes/images/service_solution.png create mode 100644 topics/kubernetes/killing_containers.md create mode 100644 topics/kubernetes/pods_01.md create mode 100644 topics/kubernetes/replicaset_01.md create mode 100644 topics/kubernetes/replicaset_02.md create mode 100644 topics/kubernetes/replicaset_03.md create mode 100644 topics/kubernetes/services_01.md create mode 100644 topics/kubernetes/solutions/killing_containers.md create mode 100644 topics/kubernetes/solutions/pods_01_solution.md create mode 100644 topics/kubernetes/solutions/replicaset_01_solution.md create mode 100644 topics/kubernetes/solutions/replicaset_02_solution.md create mode 100644 topics/kubernetes/solutions/replicaset_03_solution.md create mode 100644 topics/kubernetes/solutions/services_01_solution.md diff --git a/topics/kubernetes/CKA.md b/topics/kubernetes/CKA.md new file mode 100644 index 0000000..90ecf8f --- /dev/null +++ b/topics/kubernetes/CKA.md @@ -0,0 +1,868 @@ +# CKA (Certified Kubernetes Administrator) + +![CKA](https://imgur.com/TAOSiBB.png) + +- [CKA (Certified Kubernetes Administrator)](#cka-certified-kubernetes-administrator) + - [Setup](#setup) + - [Pods](#pods) + - [Troubleshooting Pods](#troubleshooting-pods) + - [Namespaces](#namespaces) + - [Nodes](#nodes) + - [Services](#services) + - [ReplicaSets](#replicasets) + - [Troubleshooting ReplicaSets](#troubleshooting-replicasets) + - [Deployments](#deployments) + - [Troubleshooting Deployments](#troubleshooting-deployments) + - [Scheduler](#scheduler) + - [Node Affinity](#node-affinity) + - [Labels and Selectors](#labels-and-selectors) + - [Node Selector](#node-selector) + - [Taints](#taints) + - [Resources Limits](#resources-limits) + - [Monitoring](#monitoring) + - [Scheduler](#scheduler-1) + +## Setup + +* Set up Kubernetes cluster. Use on of the following + 1. Minikube for local free & simple cluster + 2. Managed Cluster (EKS, GKE, AKS) + +* Set aliases + +``` +alias k=kubectl +alias kd=kubectl delete +alias kds=kubectl describe +alias ke=kubectl edit +alias kr=kubectl run +alias kg=kubectl get +``` + +## Pods + +
+Run a command to view all the pods in the current namespace
+ +`kubectl get pods` + +Note: create an alias (`alias k=kubectl`) and get used to `k get po` +
+ +
+Run a pod called "nginx-test" using the "nginx" image
+ +`k run nginx-test --image=nginx` +
+ +
+Assuming you have a Pod called "nginx-test", how to remove it?
+ +`k delete nginx-test` +
+ +
+In what namespace the etcd pod is running? list the pods in that namespace
+ +`k get po -n kube-system` + +Let's say you didn't know in what namespace it is. You could then run `k get po -A | grep etc` to find the Pod and see in what namespace it resides. +
+ +
+List pods from all namespaces
+ +`k get po -A` + +The long version would be `kubectl get pods --all-namespaces`. +
+ +
+Write a YAML of a Pod with two containers and use the YAML file to create the Pod (use whatever images you prefer)
+ +``` +cat > pod.yaml < pod.yaml`. If you ask yourself "how am I supposed to remember this long command" time to change attitude ;) +
+ +
+Create a YAML of a Pod without actually running the Pod with the kubectl command (use whatever image you prefer)
+ +`k run some-pod -o yaml --image nginx-unprivileged --dry-run=client > pod.yaml` +
+ +
+How to test a manifest is valid?
+ +with `--dry-run` flag which will not actually create it, but it will test it and you can find this way any syntax issues. + +`k create -f YAML_FILE --dry-run` +
+ +
+How to check which image a certain Pod is using?
+ +`k describe po | grep -i image` +
+ +
+How to check how many containers run in single Pod?
+ +`k get po POD_NAME` and see the number under "READY" column. + +You can also run `k describe po POD_NAME` +
+ +
+Run a Pod called "remo" with the the latest redis image and the label 'year=2017'
+ +`k run remo --image=redis:latest -l year=2017` +
+ +
+List pods and their labels
+ +`k get po --show-labels` +
+ +
+Delete a Pod called "nm"
+ +`k delete po nm` +
+ +
+List all the pods with the label "env=prod"
+ +`k get po -l env=prod` + +To count them: `k get po -l env=prod --no-headers | wc -l` +
+ +
+Create a static pod with the image python that runs the command sleep 2017
+ +First change to the directory tracked by kubelet for creating static pod: `cd /etc/kubernetes/manifests` (you can verify path by reading kubelet conf file) + +Now create the definition/manifest in that directory +`k run some-pod --image=python --command sleep 2017 --restart=Never --dry-run=client -o yaml > static-pod.yaml` +
+ +
+Describe how would you delete a static Pod +
+ +Locate the static Pods directory (look at `staticPodPath` in kubelet configuration file). + +Go to that directory and remove the manifest/definition of the staic Pod (`rm /`) +
+ +### Troubleshooting Pods + +
+You try to run a Pod but see the status "CrashLoopBackOff". What does it means? How to identify the issue?
+ +The container failed to run (due to different reasons) and Kubernetes tries to run the Pod again after some delay (= BackOff time). + +Some reasons for it to fail: + - Misconfiguration - misspelling, non supported value, etc. + - Resource not available - nodes are down, PV not mounted, etc. + +Some ways to debug: + +1. `kubectl describe pod POD_NAME` + 1. Focus on `State` (which should be Waiting, CrashLoopBackOff) and `Last State` which should tell what happened before (as in why it failed) +2. Run `kubectl logs mypod` + 1. This should provide an accurate output of + 2. For specific container, you can add `-c CONTAINER_NAME` +3. If you still have no idea why it failed, try `kubectl get events` +4. +
+ +
+What the error ImagePullBackOff means?
+ +Most likely you didn't write correctly the name of the image you try to pull and run. Or perhaps it doesn't exists in the registry. + +You can confirm with `kubectl describe po POD_NAME` +
+ +
+How to check on which node a certain Pod is running?
+ +`k get po POD_NAME -o wide` +
+ +
+Run the following command: kubectl run ohno --image=sheris. Did it work? why not? fix it without removing the Pod and using any image you would like
+ +Because there is no such image `sheris`. At least for now :) + +To fix it, run `kubectl edit ohno` and modify the following line `- image: sheris` to `- image: redis` or any other image you prefer. +
+ +
+You try to run a Pod but it's in "Pending" state. What might be the reason?
+ +One possible reason is that the scheduler which supposed to schedule Pods on nodes, is not running. To verify it, you can run `kubectl get po -A | grep scheduler` or check directly in `kube-system` namespace. +
+ +
+How to view the logs of a container running in a Pod?
+ +`k logs POD_NAME` +
+ +
+There are two containers inside a Pod called "some-pod". What will happen if you run kubectl logs some-pod
+ +It won't work because there are two containers inside the Pod and you need to specify one of them with `kubectl logs POD_NAME -c CONTAINER_NAME` +
+ +## Namespaces + +
+List all the namespaces
+ +`k get ns` +
+ +
+Create a namespace called 'alle'
+ +`k create ns alle` +
+ +
+Check how many namespaces are there
+ +`k get ns --no-headers | wc -l` +
+ +
+Check how many pods exist in the "dev" namespace
+ +`k get po -n dev` +
+ +
+Create a pod called "kartos" in the namespace dev. The pod should be using the "redis" image.
+ +If the namespace doesn't exist already: `k create ns dev` + +`k run kratos --image=redis -n dev` +
+ +
+You are looking for a Pod called "atreus". How to check in which namespace it runs?
+ +`k get po -A | grep atreus` +
+ +## Nodes + +
+Run a command to view all nodes of the cluster
+ +`kubectl get nodes` + +Note: create an alias (`alias k=kubectl`) and get used to `k get no` +
+ +
+Create a list of all nodes in JSON format and store it in a file called "some_nodes.json"
+ +`k get nodes -o json > some_nodes.json` +
+ +
+Check what labels one of your nodes in the cluster has
+ +`k get no minikube --show-labels` +
+ +## Services + +
+Check how many services are running in the current namespace
+ +`k get svc` +
+ +
+Create an internal service called "sevi" to expose the app 'web' on port 1991
+ +`kubectl expose pod web --port=1991 --name=sevi` +
+ +
+How to reference by name a service called "app-service" within the same namespace?
+ +app-service +
+ +
+How to check the TargetPort of a service?
+ +`k describe svc ` +
+ +
+How to check what endpoints the svc has?
+ +`k describe svc ` +
+ +
+How to reference by name a service called "app-service" within a different namespace, called "dev"?
+ +app-service.dev.svc.cluster.local +
+ +
+Assume you have a deployment running and you need to create a Service for exposing the pods. This is what is required/known: + +* Deployment name: jabulik +* Target port: 8080 +* Service type: NodePort +* Selector: jabulik-app +* Port: 8080 +
+ +`kubectl expose deployment jabulik --name=jabulik-service --target-port=8080 --type=NodePort --port=8080 --dry-run=client -o yaml -> svc.yaml` + +`vi svc.yaml` (make sure selector is set to `jabulik-app`) + +`k apply -f svc.yaml` +
+ +## ReplicaSets + +
+How to check how many replicasets defined in the current namespace?
+ +`k get rs` +
+ +
+You have a replica set defined to run 3 Pods. You removed one of these 3 pods. What will happen next? how many Pods will there be?
+ +There will still be 3 Pods running theoretically because the goal of the replica set is to ensure that. so if you delete one or more Pods, it will run additional Pods so there are always 3 Pods. +
+ +
+How to check which container image was used as part of replica set called "repli"?
+ +`k describe rs repli | grep -i image` +
+ +
+How to check how many Pods are ready as part of a replica set called "repli"?
+ +`k describe rs repli | grep -i "Pods Status"` +
+ +
+How to delete a replica set called "rori"?
+ +`k delete rs rori` +
+ +
+How to modify a replica set called "rori" to use a different image?
+ +`k edis rs rori` +
+ +
+Scale up a replica set called "rori" to run 5 Pods instead of 2
+ +`k scale rs rori --replicas=5` +
+ +
+Scale down a replica set called "rori" to run 1 Pod instead of 5
+ +`k scale rs rori --replicas=1` +
+ +### Troubleshooting ReplicaSets + +
+Fix the following ReplicaSet definition + +```yaml +apiVersion: apps/v1 +kind: ReplicaCet +metadata: + name: redis + labels: + app: redis + tier: cache +spec: + selector: + matchLabels: + tier: cache + template: + metadata: + labels: + tier: cachy + spec: + containers: + - name: redis + image: redis +``` +
+ +kind should be ReplicaSet and not ReplicaCet :) + +
+ +
+Fix the following ReplicaSet definition + +```yaml +apiVersion: apps/v1 +kind: ReplicaSet +metadata: + name: redis + labels: + app: redis + tier: cache +spec: + selector: + matchLabels: + tier: cache + template: + metadata: + labels: + tier: cachy + spec: + containers: + - name: redis + image: redis +``` +
+ +The selector doesn't match the label (cache vs cachy). To solve it, fix cachy so it's cache instead. + +
+ +## Deployments + +
+How to list all the deployments in the current namespace?
+ +`k get deploy` + +
+ +
+How to check which image a certain Deployment is using?
+ +`k describe deploy | grep image` + +
+ +
+Create a file definition/manifest of a deployment called "dep", with 3 replicas that uses the image 'redis'
+ +`k create deploy dep -o yaml --image=redis --dry-run=client --replicas 3 > deployment.yaml ` + +
+ +
+Remove the deployment `depdep`
+ +`k delete deploy depdep` + +
+ +
+Create a deployment called "pluck" using the image "redis" and make sure it runs 5 replicas
+ +`kubectl create deployment pluck --image=redis --replicas=5` + +
+ +
+Create a deployment with the following properties: + +* called "blufer" +* using the image "python" +* runs 3 replicas +* all pods will be placed on a node that has the label "blufer" +
+ +`kubectl create deployment blufer --image=python --replicas=3 -o yaml --dry-run=client > deployment.yaml` + +Add the following section (`vi deployment.yaml`): + +``` +spec: + affinity: + nodeAffinity: + requiredDuringSchedlingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: blufer + operator: Exists +``` + +`kubectl apply -f deployment.yaml` +
+ +### Troubleshooting Deployments + +
+Fix the following deployment manifest + +```yaml +apiVersion: apps/v1 +kind: Deploy +metadata: + creationTimestamp: null + labels: + app: dep + name: dep +spec: + replicas: 3 + selector: + matchLabels: + app: dep + strategy: {} + template: + metadata: + creationTimestamp: null + labels: + app: dep + spec: + containers: + - image: redis + name: redis + resources: {} +status: {} +``` +
+ +Change `kind: Deploy` to `kind: Deployment` +
+ +
+Fix the following deployment manifest + +```yaml +apiVersion: apps/v1 +kind: Deployment +metadata: + creationTimestamp: null + labels: + app: dep + name: dep +spec: + replicas: 3 + selector: + matchLabels: + app: depdep + strategy: {} + template: + metadata: + creationTimestamp: null + labels: + app: dep + spec: + containers: + - image: redis + name: redis + resources: {} +status: {} +``` +
+ +The selector doesn't match the label (dep vs depdep). To solve it, fix depdep so it's dep instead. +
+ +## Scheduler + +
+How to schedule a pod on a node called "node1"?
+ +`k run some-pod --image=redix -o yaml --dry-run=client > pod.yaml` + +`vi pod.yaml` and add: + +``` +spec: + nodeName: node1 +``` + +`k apply -f pod.yaml` + +Note: if you don't have a node1 in your cluster the Pod will be stuck on "Pending" state. +
+ +### Node Affinity + +
+Using node affinity, set a Pod to schedule on a node where the key is "region" and value is either "asia" or "emea"
+ +`vi pod.yaml` + +```yaml +affinity: + nodeAffinity: + requiredDuringSchedlingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: region + operator: In + values: + - asia + - emea +``` +
+ +
+Using node affinity, set a Pod to never schedule on a node where the key is "region" and value is "neverland"
+ +`vi pod.yaml` + +```yaml +affinity: + nodeAffinity: + requiredDuringSchedlingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: region + operator: NotIn + values: + - neverland +``` +
+ +## Labels and Selectors + +
+How to list all the Pods with the label "app=web"?
+ +`k get po -l app=web` +
+ +
+How to list all objects labeled as "env=staging"?
+ +`k get all -l env=staging` +
+ +
+How to list all deployments from "env=prod" and "type=web"?
+ +`k get deploy -l env=prod,type=web` +
+ +### Node Selector + +
+Apply the label "hw=max" on one of the nodes in your cluster
+ +`kubectl label nodes some-node hw=max` + +
+ +
+Create and run a Pod called `some-pod` with the image `redis` and configure it to use the selector `hw=max`
+ +``` +kubectl run some-pod --image=redis --dry-run=client -o yaml > pod.yaml + +vi pod.yaml + +spec: + nodeSelector: + hw: max + +kubectl apply -f pod.yaml +``` + +
+ +
+Explain why node selectors might be limited
+ +Assume you would like to run your Pod on all the nodes with with either `hw` set to max or to min, instead of just max. This is not possible with nodeSelectors which are quite simplified and this is where you might want to consider `node affinity`. +
+ +## Taints + +
+Check if there are taints on node "master"
+ +`k describe no master | grep -i taints` +
+ +
+Create a taint on one of the nodes in your cluster with key of "app" and value of "web" and effect of "NoSchedule". Verify it was applied
+ +`k taint node minikube app=web:NoSchedule` + +`k describe no minikube | grep -i taints` +
+ +
+You applied a taint with k taint node minikube app=web:NoSchedule on the only node in your cluster and then executed kubectl run some-pod --image=redis. What will happen?
+ +The Pod will remain in "Pending" status due to the only node in the cluster having a taint of "app=web". +
+ +
+You applied a taint with k taint node minikube app=web:NoSchedule on the only node in your cluster and then executed kubectl run some-pod --image=redis but the Pod is in pending state. How to fix it?
+ +`kubectl edit po some-pod` and add the following + +``` + - effect: NoSchedule + key: app + operator: Equal + value: web +``` + +Exit and save. The pod should be in Running state now. +
+ +
+Remove an existing taint from one of the nodes in your cluster
+ +`k taint node minikube app=web:NoSchedule-` +
+ +## Resources Limits + +
+Check if there are any limits on one of the pods in your cluster
+ +`kubectl describe po | grep -i limits` +
+ +
+Run a pod called "yay" with the image "python" and resources request of 64Mi memory and 250m CPU
+ +`kubectl run yay --image=python --dry-run=client -o yaml > pod.yaml` + +`vi pod.yaml` + +``` +spec: + containers: + - image: python + imagePullPolicy: Always + name: yay + resources: + requests: + cpu: 250m + memory: 64Mi +``` + +`kubectl apply -f pod.yaml` +
+ +
+Run a pod called "yay2" with the image "python". Make sure it has resources request of 64Mi memory and 250m CPU and the limits are 128Mi memory and 500m CPU
+ +`kubectl run yay2 --image=python --dry-run=client -o yaml > pod.yaml` + +`vi pod.yaml` + +``` +spec: + containers: + - image: python + imagePullPolicy: Always + name: yay2 + resources: + limits: + cpu: 500m + memory: 128Mi + requests: + cpu: 250m + memory: 64Mi +``` + +`kubectl apply -f pod.yaml` +
+ +## Monitoring + +
+Deploy metrics-server
+ +`kubectl apply -f https://github.com/kubernetes-sigs/metrics-server/releases/latest/download/components.yaml` +
+ +
+Using metrics-server, view the following: + +* top performing nodes in the cluster +* top performing Pods +
+ +* top nodes: `kubectl top nodes` +* top pods: `kubectl top pods` + +
+ +## Scheduler + +
+Can you deploy multiple schedulers?
+ +Yes, it is possible. You can run another pod with a command similar to: + +``` +spec: + containers: + - command: + - kube-scheduler + - --address=127.0.0.1 + - --leader-elect=true + - --scheduler-name=some-custom-scheduler +... +``` +
+ +
+Assuming you have multiple schedulers, how to know which scheduler was used for a given Pod?
+ +Running `kubectl get events` you can see which scheduler was used. +
+ +
+You want to run a new Pod and you would like it to be scheduled by a custom scheduler. How to achieve it?
+ +Add the following to the spec of the Pod: + +``` +spec: + schedulerName: some-custom-scheduler +``` +
+ + +# Hit the Star! ⭐ +***If you are planning to use this repo for learning, please hit the star. Thanks!*** + +#### Author by [Harshhaa Reddy](https://github.com/NotHarshhaa) \ No newline at end of file diff --git a/topics/kubernetes/README.md b/topics/kubernetes/README.md new file mode 100644 index 0000000..1a1f8bd --- /dev/null +++ b/topics/kubernetes/README.md @@ -0,0 +1,3157 @@ +# Kubernetes + +![Kubernetes](https://imgur.com/ULpBgBC.png) + + + +What's your goal? + +* I would like to prepare for CKA certification + * See [CKA](CKA.md) page +* I would like to learn Kubernetes by practicing both theoritcal and practical material + * Solve [exercises](#kubernetes-exercises) + * Solve [questions](#kubernetes-questions) +* I would like to learn practical Kubernetes + * Solve [exercises](#kubernetes-exercises) + +- [Kubernetes](#kubernetes) + - [Kubernetes Exercises](#kubernetes-exercises) + - [Pods](#pods) + - [Service](#service) + - [ReplicaSet](#replicaset) + - [Labels and Selectors](#labels-and-selectors) + - [Scheduler](#scheduler) + - [Kustomize](#kustomize) + - [Kubernetes Questions](#kubernetes-questions) + - [Kubernetes 101](#kubernetes-101) + - [Cluster and Architecture](#cluster-and-architecture) + - [Kubelet](#kubelet) + - [Nodes Commands](#nodes-commands) + - [Pods](#pods-1) + - [Static Pods](#static-pods) + - [Pods Commands](#pods-commands) + - [Pods Troubleshooting and Debugging](#pods-troubleshooting-and-debugging) + - [Labels and Selectors](#labels-and-selectors-1) + - [Deployments](#deployments) + - [Deployments Commands](#deployments-commands) + - [Services](#services) + - [Ingress](#ingress) + - [ReplicaSets](#replicasets) + - [DaemonSet](#daemonset) + - [DaemonSet - Commands](#daemonset---commands) + - [StatefulSet](#statefulset) + - [Storage](#storage) + - [Volumes](#volumes) + - [Networking](#networking) + - [Network Policies](#network-policies) + - [etcd](#etcd) + - [Namespaces](#namespaces) + - [Namespaces - commands](#namespaces---commands) + - [Resources Quota](#resources-quota) + - [Operators](#operators) + - [Secrets](#secrets) + - [Volumes](#volumes-1) + - [Access Control](#access-control) + - [Patterns](#patterns) + - [CronJob](#cronjob) + - [Misc](#misc) + - [Gatekeeper](#gatekeeper) + - [Policy Testing](#policy-testing) + - [Helm](#helm) + - [Commands](#commands) + - [Security](#security) + - [Troubleshooting Scenarios](#troubleshooting-scenarios) + - [Istio](#istio) + - [Controllers](#controllers) + - [Scheduler](#scheduler-1) + - [Node Affinity](#node-affinity) + - [Taints](#taints) + - [Resource Limits](#resource-limits) + - [Resources Limits - Commands](#resources-limits---commands) + - [Monitoring](#monitoring) + - [Kustomize](#kustomize-1) + - [Deployment Strategies](#deployment-strategies) + - [Scenarios](#scenarios) + +## Kubernetes Exercises + +### Pods + +|Name|Topic|Objective & Instructions|Solution|Comments| +|--------|--------|------|----|----| +| My First Pod | Pods | [Exercise](pods_01.md) | [Solution](solutions/pods_01_solution.md) +| "Killing" Containers | Pods | [Exercise](killing_containers.md) | [Solution](solutions/killing_containers.md) + +### Service + +|Name|Topic|Objective & Instructions|Solution|Comments| +|--------|--------|------|----|----| +| Creating a Service | Service | [Exercise](services_01.md) | [Solution](solutions/services_01_solution.md) + +### ReplicaSet + +|Name|Topic|Objective & Instructions|Solution|Comments| +|--------|--------|------|----|----| +| Creating a ReplicaSet | ReplicaSet | [Exercise](replicaset_01.md) | [Solution](solutions/replicaset_01_solution.md) +| Operating ReplicaSets | ReplicaSet | [Exercise](replicaset_02.md) | [Solution](solutions/replicaset_02_solution.md) +| ReplicaSets Selectors | ReplicaSet | [Exercise](replicaset_03.md) | [Solution](solutions/replicaset_03_solution.md) + +### Labels and Selectors + +|Name|Topic|Objective & Instructions|Solution|Comments| +|--------|--------|------|----|----| +| Labels and Selectors 101 | Labels, Selectors | [Exercise](exercises/labels_and_selectors/exercise.md) | [Solution](exercises/labels_and_selectors/solution.md) +| Node Selectors | Labels, Selectors | [Exercise](exercises/node_selectors/exercise.md) | [Solution](exercises/node_selectors/solution.md) + + +### Scheduler + +|Name|Topic|Objective & Instructions|Solution|Comments| +|--------|--------|------|----|----| +| Taints 101 | Taints | [Exercise](exercises/taints_101/exercise.md) | [Solution](exercises/taints_101/solution.md) + +### Kustomize + +|Name|Topic|Objective & Instructions|Solution|Comments| +|--------|--------|------|----|----| +| common labels | Kustomize | [Exercise](exercises/kustomize_common_labels/exercise.md) | [Solution](exercises/kustomize_common_labels/solution.md) + +## Kubernetes Questions + +### Kubernetes 101 + +
+What is Kubernetes? Why organizations are using it?
+ +Kubernetes is an open-source system that provides users with the ability to manage, scale and deploy containerized applications. + +To understand what Kubernetes is good for, let's look at some examples: + +* You would like to run a certain application in a container on multiple different locations and sync changes across all of them, no matter where they run +* Performing updates and changes across hundreds of containers +* Handle cases where the current load requires to scale up (or down) + +
+ +
+When or why NOT to use Kubernetes?
+ + - If you manage low level infrastructure or baremetals, Kubernetes is probably not what you need or want + - If you are a small team (like less than 20 engineers) running less than a dozen of containers, Kubernetes might be an overkill (even if you need scale, rolling out updates, etc.). You might still enjoy the benefits of using managed Kubernetes, but you definitely want to think about it carefully before making a decision on whether to adopt it. + +
+ +
+What are some of Kubernetes features?
+ + - Self-Healing: Kubernetes uses health checks to monitor containers and run certain actions upon failure or other type of events, like restarting the container + - Load Balancing: Kubernetes can split and/or balance requests to applications running in the cluster, based on the state of the Pods running the application + - Operators: Kubernetes packaged applications that can use the API of the cluster to update its state and trigger actions based on events and application state changes + - Automated Rollout: Gradual updates roll out to applications and support in roll back in case anything goes wrong + - Scaling: Scaling horizontally (down and up) based on different state parameters and custom defined criteria + - Secrets: you have a mechanism for storing user names, passwords and service endpoints in a private way, where not everyone using the cluster are able to view it + +
+ +
+What Kubernetes objects are there?
+ + * Pod + * Service + * ReplicationController + * ReplicaSet + * DaemonSet + * Namespace + * ConfigMap + ... +
+ +
+What fields are mandatory with any Kubernetes object?
+ +metadata, kind and apiVersion + +
+ +
+What is kubectl?
+ +Kubectl is the Kubernetes command line tool that allows you to run commands against Kubernetes clusters. For example, you can use kubectl to deploy applications, inspect and manage cluster resources, and view logs. + +
+ +
+What Kubernetes objects do you usually use when deploying applications in Kubernetes?
+ +* Deployment - creates the Pods () and watches them +* Service: route traffic to Pods internally +* Ingress: route traffic from outside the cluster + +
+ +
+Why there is no such command in Kubernetes? kubectl get containers
+ +Becaused container is not a Kubernetes object. The smallest object unit in Kubernetes is a Pod. In a single Pod you can find one or more containers. + +
+ +
+What actions or operations you consider as best practices when it comes to Kubernetes?
+ + - Always make sure Kubernetes YAML files are valid. Applying automated checks and pipelines is recommended. + - Always specify requests and limits to prevent situation where containers are using the entire cluster memory which may lead to OOM issue + - Specify labels to logically group Pods, Deployments, etc. Use labels to identify the type of the application for example, among other things + +
+ +### Cluster and Architecture + +
+What is a Kubernetes Cluster?
+ +Red Hat Definition: "A Kubernetes cluster is a set of node machines for running containerized applications. If you’re running Kubernetes, you’re running a cluster. +At a minimum, a cluster contains a worker node and a master node." + +Read more [here](https://www.redhat.com/en/topics/containers/what-is-a-kubernetes-cluster) +
+ +
+What is a Node?
+ +A node is a virtual or a physical machine that serves as a worker for running the applications.
+It's recommended to have at least 3 nodes in a production environment. +
+ +
+What the master node is responsible for?
+ +The master coordinates all the workflows in the cluster: + +* Scheduling applications +* Managing desired state +* Rolling out new updates + +
+ +
+Describe shortly and in high-level, what happens when you run kubectl get nodes
+ +1. Your user is getting authenticated +2. Request is validated by the kube-apiserver +3. Data is retrieved from etcd +
+ +
+True or False? Every cluster must have 0 or more master nodes and at least 1 worker
+ +False. A Kubernetes cluster consists of at least 1 master and can have 0 workers (although that wouldn't be very useful...) + +
+ +
+What are the components of the master node (aka control plane)?
+ + * API Server - the Kubernetes API. All cluster components communicate through it + * Scheduler - assigns an application with a worker node it can run on + * Controller Manager - cluster maintenance (replications, node failures, etc.) + * etcd - stores cluster configuration + +
+ +
+What are the components of a worker node (aka data plane)?
+ + * Kubelet - an agent responsible for node communication with the master. + * Kube-proxy - load balancing traffic between app components + * Container runtime - the engine runs the containers (Podman, Docker, ...) + +
+ +
+Place the components on the right side of the image in the right place in the drawing
+ +

+ + +
+ +
+You are managing multiple Kubernetes clusters. How do you quickly change between the clusters using kubectl?
+ +`kubectl config use-context` +
+ +
+How do you prevent high memory usage in your Kubernetes cluster and possibly issues like memory leak and OOM?
+ +Apply requests and limits, especially on third party applications (where the uncertainty is even bigger) +
+ +
+Do you have experience with deploying a Kubernetes cluster? If so, can you describe the process in high-level?
+ +1. Create multiple instances you will use as Kubernetes nodes/workers. Create also an instance to act as the Master. The instances can be provisioned in a cloud or they can be virtual machines on bare metal hosts. +2. Provision a certificate authority that will be used to generate TLS certificates for the different components of a Kubernetes cluster (kubelet, etcd, ...) + 1. Generate a certificate and private key for the different components +3. Generate kubeconfigs so the different clients of Kubernetes can locate the API servers and authenticate. +4. Generate encryption key that will be used for encrypting the cluster data +5. Create an etcd cluster +
+ +
+Which command will list all the object types in a cluster?
+ +`kubectl api-resources` +
+ +
+What kubectl get componentstatus does?
+ +Outputs the status of each of the control plane components. +
+ +#### Kubelet + +
+What happens to running pods if if you stop Kubelet on the worker nodes?
+ +
+ +#### Nodes Commands + +
+Run a command to view all nodes of the cluster
+ +`kubectl get nodes` + +Note: You might want to create an alias (`alias k=kubectl`) and get used to `k get no` +
+ +
+Create a list of all nodes in JSON format and store it in a file called "some_nodes.json"
+ +`k get nodes -o json > some_nodes.json` +
+ +
+Check what labels one of your nodes in the cluster has
+ +`k get no minikube --show-labels` +
+ +### Pods + +
+Explain what is a Pod
+ +A Pod is a group of one or more containers, with shared storage and network resources, and a specification for how to run the containers. + +Pods are the smallest deployable units of computing that you can create and manage in Kubernetes. + +
+ +
+Deploy a pod called "my-pod" using the nginx:alpine image
+ +`kubectl run my-pod --image=nginx:alpine` + +If you are a Kubernetes beginner you should know that this is not a common way to run Pods. The common way is to run a Deployment which in turn runs Pod(s). + +In addition, Pods and/or Deployments are usually defined in files rather than executed directly using only the CLI arguments. +
+ +
+What are your thoughts on "Pods are not meant to be created directly"?
+ +Pods are usually indeed not created directly. You'll notice that Pods are usually created as part of another entities such as Deployments or ReplicaSets. + +If a Pod dies, Kubernetes will not bring it back. This is why it's more useful for example to define ReplicaSets that will make sure that a given number of Pods will always run, even after a certain Pod dies. +
+ +
+How many containers can a pod contain?
+ +A pod can include multiple containers but in most cases it would probably be one container per pod. + +There are some patterns where it makes to run more than one container like the "side-car" pattern where you might want to perform logging or some other operation that is executed by another container running with your app container in the same Pod. +
+ +
+What use cases exist for running multiple containers in a single pod?
+ +A web application with separate (= in their own containers) logging and monitoring components/adapters is one examples.
+A CI/CD pipeline (using Tekton for example) can run multiple containers in one Pod if a Task contains multiple commands. +
+ +
+What are the possible Pod phases?
+ + * Running - The Pod bound to a node and at least one container is running + * Failed/Error - At least one container in the Pod terminated with a failure + * Succeeded - Every container in the Pod terminated with success + * Unknown - Pod's state could not be obtained + * Pending - Containers are not yet running (Perhaps images are still being downloaded or the pod wasn't scheduled yet) +
+ +
+True or False? By default, pods are isolated. This means they are unable to receive traffic from any source
+ +False. By default, pods are non-isolated = pods accept traffic from any source. +
+ +
+True or False? The "Pending" phase means the Pod was not yet accepted by the Kubernetes cluster so the scheduler can't run it unless it's accepted
+ +False. "Pending" is after the Pod was accepted by the cluster, but the container can't run for different reasons like images not yet downloaded. +
+ +
+True or False? A single Pod can be split across multiple nodes
+ +False. A single Pod can run on a single node. +
+ +
+You run a pod and you see the status ContainerCreating
+
+ +
+True or False? A volume defined in Pod can be accessed by all the containers of that Pod
+ +True. +
+ +
+What happens when you run a Pod with kubectl?
+ +1. Kubectl sends a request to the API server (kube-apiserver) to create the Pod + 1. In the the process the user gets authenticated and the request is being validated. + 2. etcd is being updated with the data +2. The Scheduler detects that there is an unassigned Pod by monitoring the API server (kube-apiserver) +3. The Scheduler chooses a node to assign the Pod to + 1. etcd is being updated with the information +4. The Scheduler updates the API server about which node it chose +5. Kubelet (which also monitors the API server) notices there is a Pod assigned to the same node on which it runs and that Pod isn't running +6. Kubelet sends request to the container engine (e.g. Docker) to create and run the containers +7. An update is sent by Kubelet to the API server (notifying it that the Pod is running) + 1. etcd is being updated by the API server again +
+ +
+How to confirm a container is running after running the command kubectl run web --image nginxinc/nginx-unprivileged
+ +* When you run `kubectl describe pods ` it will tell whether the container is running: +`Status: Running` +* Run a command inside the container: `kubectl exec web -- ls` +
+ +
+After running kubectl run database --image mongo you see the status is "CrashLoopBackOff". What could possibly went wrong and what do you do to confirm?
+ +"CrashLoopBackOff" means the Pod is starting, crashing, starting...and so it repeats itself.
+There are many different reasons to get this error - lack of permissions, init-container misconfiguration, persistent volume connection issue, etc. + +One of the ways to check why it happened it to run `kubectl describe po ` and having a look at the exit code + +``` + Last State: Terminated + Reason: Error + Exit Code: 100 +``` + +Another way to check what's going on, is to run `kubectl logs `. This will provide us with the logs from the containers running in that Pod. +
+ +
+Explain the purpose of the following lines + +``` +livenessProbe: + exec: + command: + - cat + - /appStatus + initialDelaySeconds: 10 + periodSeconds: 5 +``` +
+ +These lines make use of `liveness probe`. It's used to restart a container when it reaches a non-desired state.
+In this case, if the command `cat /appStatus` fails, Kubernetes will kill the container and will apply the restart policy. The `initialDelaySeconds: 10` means that Kubelet will wait 10 seconds before running the command/probe for the first time. From that point on, it will run it every 5 seconds, as defined with `periodSeconds` +
+ +
+Explain the purpose of the following lines + +``` +readinessProbe: + tcpSocket: + port: 2017 + initialDelaySeconds: 15 + periodSeconds: 20 +``` +
+ +They define a readiness probe where the Pod will not be marked as "Ready" before it will be possible to connect to port 2017 of the container. The first check/probe will start after 15 seconds from the moment the container started to run and will continue to run the check/probe every 20 seconds until it will manage to connect to the defined port. +
+ +
+What does the "ErrImagePull" status of a Pod means?
+ +It wasn't able to pull the image specified for running the container(s). This can happen if the client didn't authenticated for example.
+More details can be obtained with `kubectl describe po `. +
+ +
+What happens when you delete a Pod?
+ +1. The `TERM` signal is sent to kill the main processes inside the containers of the given Pod +2. Each container is given a period of 30 seconds to shut down the processes gracefully +3. If the grace period expires, the `KILL` signal is used to kill the processes forcefully and the containers as well +
+ +
+Explain liveness probes
+ +Liveness probes is a useful mechanism used for restarting the container when a certain check/probe, the user has defined, fails.
+For example, the user can define that the command `cat /app/status` will run every X seconds and the moment this command fails, the container will be restarted. + +You can read more about it in [kubernetes.io](https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes) +
+ +
+Explain readiness probes
+ +readiness probes used by Kubelet to know when a container is ready to start running, accepting traffic.
+For example, a readiness probe can be to connect port 8080 on a container. Once Kubelet manages to connect it, the Pod is marked as ready + +You can read more about it in [kubernetes.io](https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes) +
+ +
+How readiness probe status affect Services when they are combined?
+ +Only containers whose state set to Success will be able to receive requests sent to the Service. +
+ +
+Why it's common to have only one container per Pod in most cases?
+ +One reason is that it makes it harder to scale when you need to scale only one of the containers in a given Pod. +
+ +
+True or False? Once a Pod is assisgned to a worker node, it will only run on that node, even if it fails at some point and spins up a new Pod
+ +True. +
+ +
+True or False? Each Pod, when created, gets its own public IP address
+ +False. Each Pod gets an IP address but an internal one and not publicly accessible. + +To make a Pod externally accessible, we need to use an object called Service in Kubernetes. +
+ +#### Static Pods + +
+What are Static Pods?
+ +[Kubernetes.io](https://kubernetes.io/docs/tasks/configure-pod-container/static-pod/): "Static Pods are managed directly by the kubelet daemon on a specific node, without the API server observing them. Unlike Pods that are managed by the control plane (for example, a Deployment); instead, the kubelet watches each static Pod (and restarts it if it fails)." +
+ +
+True or False? The same as there are "Static Pods" there are other static resources like "deployments" and "replicasets"
+ +False. +
+ +
+What are some use cases for using Static Pods?
+ +One clear use case is running Control Plane Pods - running Pods such as kube-apiserver, scheduler, etc. These should run and operate regardless of whether some components of the cluster work or not and they should run on specific nodes of the cluster. +
+ +
+How to identify which Pods are Static Pods?
+ +The suffix of the Pods is the same as the name of the nodes on which they are running +TODO: check if it's always the case. +
+ +
+Which of the following is not a static pod?: + +* kube-scheduler +* kube-proxy +* kube-apiserver +
+ +kube-proxy - it's a DaemonSet (since it has to be presented on every node in the cluster). There is no one specific node on which it has to run. +
+ +
+Where static Pods manifests are located?
+ +Most of the time it's in /etc/kubernetes/manifests but you can verify with `grep -i static /var/lib/kubelet/config.yaml` to locate the value of `statisPodsPath`. + +It might be that your config is in different path. To verify run `ps -ef | grep kubelet` and see what is the value of --config argument of the process `/usr/bin/kubelet` + +The key itself for defining the path of static Pods is `staticPodPath`. So if your config is in `/var/lib/kubelet/config.yaml` you can run `grep staticPodPath /var/lib/kubelet/config.yaml`. +
+ +
+Describe how would you delete a static Pod +
+ +Locate the static Pods directory (look at `staticPodPath` in kubelet configuration file). + +Go to that directory and remove the manifest/definition of the staic Pod (`rm /`) +
+ +#### Pods Commands + +
+How to check to which worker node the pods were scheduled to? In other words, how to check on which node a certain Pod is running?
+ +`kubectl get pods -o wide` +
+ +
+How to delete a pod?
+ +`kubectl delete pod pod_name` +
+ +
+List all the pods with the label "env=prod"
+ +`k get po -l env=prod` + +To count them: `k get po -l env=prod --no-headers | wc -l` +
+ +
+How to list the pods in the current namespace?
+ +`kubectl get po` +
+ +
+How view all the pods running in all the namespaces?
+ +`kubectl get pods --all-namespaces` +
+ +#### Pods Troubleshooting and Debugging + +
+You try to run a Pod but it's in "Pending" state. What might be the reason?
+ +One possible reason is that the scheduler which supposed to schedule Pods on nodes, is not running. To verify it, you can run `kubectl get po -A | grep scheduler` or check directly in `kube-system` namespace. +
+ +
+What kubectl logs [pod-name] command does?
+ +Prints the logs for a container in a pod. +
+ +
+What kubectl describe pod [pod name] does? command does?
+ +Show details of a specific resource or group of resources. +
+ +
+Create a static pod with the image python that runs the command sleep 2017
+ +First change to the directory tracked by kubelet for creating static pod: `cd /etc/kubernetes/manifests` (you can verify path by reading kubelet conf file) + +Now create the definition/manifest in that directory +`k run some-pod --image=python --command sleep 2017 --restart=Never --dry-run=client -o yaml > statuc-pod.yaml` +
+ +### Labels and Selectors + +
+Explain Labels
+ +[Kubernetes.io](https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/): "Labels are key/value pairs that are attached to objects, such as pods. Labels are intended to be used to specify identifying attributes of objects that are meaningful and relevant to users, but do not directly imply semantics to the core system. Labels can be used to organize and to select subsets of objects. Labels can be attached to objects at creation time and subsequently added and modified at any time. Each object can have a set of key/value labels defined. Each Key must be unique for a given object." +
+ +
+Explain selectors
+ +[Kubernetes.io](https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors): "Unlike names and UIDs, labels do not provide uniqueness. In general, we expect many objects to carry the same label(s). + +Via a label selector, the client/user can identify a set of objects. The label selector is the core grouping primitive in Kubernetes. + +The API currently supports two types of selectors: equality-based and set-based. A label selector can be made of multiple requirements which are comma-separated. In the case of multiple requirements, all must be satisfied so the comma separator acts as a logical AND (&&) operator." +
+ +
+Provide some actual examples of how labels are used
+ +* Can be used by the scheduler to place certain Pods (with certain labels) on specific nodes +* Used by replicasets to track pods which have to be scaled +
+ +
+What are Annotations?
+ +[Kubernetes.io](https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/): "You can use Kubernetes annotations to attach arbitrary non-identifying metadata to objects. Clients such as tools and libraries can retrieve this metadata." +
+ +
+How annotations different from labels?
+ +[Kuberenets.io](Labels can be used to select objects and to find collections of objects that satisfy certain conditions. In contrast, annotations are not used to identify and select objects. The metadata in an annotation can be small or large, structured or unstructured, and can include characters not permitted by labels.): "Labels can be used to select objects and to find collections of objects that satisfy certain conditions. In contrast, annotations are not used to identify and select objects. The metadata in an annotation can be small or large, structured or unstructured, and can include characters not permitted by labels." +
+ +
+How to view the logs of a container running in a Pod?
+ +`k logs POD_NAME` +
+ +
+There are two containers inside a Pod called "some-pod". What will happen if you run kubectl logs some-pod
+ +It won't work because there are two containers inside the Pod and you need to specify one of them with `kubectl logs POD_NAME -c CONTAINER_NAME` +
+ +### Deployments + +
+What is a "Deployment" in Kubernetes?
+ +A Kubernetes Deployment is used to tell Kubernetes how to create or modify instances of the pods that hold a containerized application. +Deployments can scale the number of replica pods, enable rollout of updated code in a controlled manner, or roll back to an earlier deployment version if necessary. + +A Deployment is a declarative statement for the desired state for Pods and Replica Sets. +
+ +
+How to create a deployment with the image "nginx:alpine"?
+ +`kubectl create deployment my_first_deployment --image=nginx:alpine` + +OR + +``` +cat << EOF | kubectl create -f - +apiVersion: v1 +kind: Pod +metadata: + name: nginx +spec: + containers: + - name: nginx + image: nginx:alpine +EOF +``` +
+ +
+How to verify a deployment was created?
+ +`kubectl get deployments` or `kubectl get deploy` + +This command lists all the Deployment objects created and exist in the cluster. It doesn't mean the deployments are readt and running. This can be checked with the "READY" and "AVAILABLE" columns. +
+ +
+How to edit a deployment?
+ +`kubectl edit deployment ` +
+ +
+What happens after you edit a deployment and change the image?
+ +The pod will terminate and another, new pod, will be created. + +Also, when looking at the replicaset, you'll see the old replica doesn't have any pods and a new replicaset is created. +
+ +
+How to delete a deployment?
+ +One way is by specifying the deployment name: `kubectl delete deployment [deployment_name]` + +Another way is using the deployment configuration file: `kubectl delete -f deployment.yaml` +
+ +
+What happens when you delete a deployment?
+ +The pod related to the deployment will terminate and the replicaset will be removed. +
+ +
+What happens behind the scenes when you create a Deployment object?
+ +The following occurs when you run `kubectl create deployment some_deployment --image=nginx` + +1. HTTP request sent to kubernetes API server on the cluster to create a new deployment +2. A new Pod object is created and scheduled to one of the workers nodes +3. Kublet on the worker node notices the new Pod and instructs the Container runtime engine to pull the image from the registry +4. A new container is created using the image that was just pulled +
+ +
+How make an app accessible on private or external network?
+ +Using a Service. +
+ +
+Can you use a Deployment for stateful applications?
+
+ +
+Fix the following deployment manifest + +```yaml +apiVersion: apps/v1 +kind: Deploy +metadata: + creationTimestamp: null + labels: + app: dep + name: dep +spec: + replicas: 3 + selector: + matchLabels: + app: dep + strategy: {} + template: + metadata: + creationTimestamp: null + labels: + app: dep + spec: + containers: + - image: redis + name: redis + resources: {} +status: {} +``` +
+ +Change `kind: Deploy` to `kind: Deployment` +
+ +
+Fix the following deployment manifest + +```yaml +apiVersion: apps/v1 +kind: Deployment +metadata: + creationTimestamp: null + labels: + app: dep + name: dep +spec: + replicas: 3 + selector: + matchLabels: + app: depdep + strategy: {} + template: + metadata: + creationTimestamp: null + labels: + app: dep + spec: + containers: + - image: redis + name: redis + resources: {} +status: {} +``` +
+ +The selector doesn't match the label (dep vs depdep). To solve it, fix depdep so it's dep instead. +
+ +#### Deployments Commands + +
+Create a file definition/manifest of a deployment called "dep", with 3 replicas that uses the image 'redis'
+ +`k create deploy dep -o yaml --image=redis --dry-run=client --replicas 3 > deployment.yaml ` + +
+ +
+Delete the deployment `depdep`
+ +`k delete deploy depdep` + +
+ +
+Create a deployment called "pluck" using the image "redis" and make sure it runs 5 replicas
+ +`kubectl create deployment pluck --image=redis` + +`kubectl scale deployment pluck --replicas=5` +
+ +
+Create a deployment with the following properties: + +* called "blufer" +* using the image "python" +* runs 3 replicas +* all pods will be placed on a node that has the label "blufer" +
+ +`kubectl create deployment blufer --image=python --replicas=3 -o yaml --dry-run=client > deployment.yaml` + +Add the following section (`vi deployment.yaml`): + +``` +spec: + affinity: + nodeAffinity: + requiredDuringSchedlingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: blufer + operator: Exists +``` + +`kubectl apply -f deployment.yaml` +
+ +### Services + +
+What is a Service in Kubernetes?
+ +"An abstract way to expose an application running on a set of Pods as a network service." - read more [here](https://kubernetes.io/docs/concepts/services-networking/service) + +In simpler words, it allows you to add an internal or external connectivity to a certain application running in a container. +
+ +
+Place the components in the right placeholders in regards to Kubernetes service
+ +

+ + + +
+ + +
+How to create a service for an existing deployment called "alle" on port 8080 so the Pod(s) accessible via a Load Balancer?
+ +The imperative way: + +`kubectl expose deployment alle --type=LoadBalancer --port 8080` +
+ +
+True or False? The lifecycle of Pods and Services isn't connected so when a Pod dies, the Service still stays
+ +True +
+ +
+After creating a service, how to check it was created?
+ +`kubectl get svc` +
+ +
+What's the default Service type?
+ +ClusterIP - used for internal communication. +
+ +
+What Service types are there?
+ +* ClusterIP +* NodePort +* LoadBalancer +* ExternalName + +More on this topic [here](https://kubernetes.io/docs/concepts/services-networking/service/#publishing-services-service-types) +
+ +
+How Service and Deployment are connected?
+ +The truth is they aren't connected. Service points to Pod(s) directly, without connecting to the Deployment in any way. +
+ +
+What are important steps in defining/adding a Service?
+ +1. Making sure that targetPort of the Service is matching the containerPort of the Pod +2. Making sure that selector matches at least one of the Pod's labels + +
+ +
+What is the default service type in Kubernetes and what is it used for?
+ +The default is ClusterIP and it's used for exposing a port internally. It's useful when you want to enable internal communication between Pods and prevent any external access. + +
+ +
+How to get information on a certain service?
+ +`kubctl describe service ` + +It's more common to use `kubectl describe svc ...` + +
+ +
+What the following command does? + +``` +kubectl expose rs some-replicaset --name=replicaset-svc --target-port=2017 --type=NodePort +``` +
+ +It exposes a ReplicaSet by creating a service called 'replicaset-svc'. The exposed port is 2017 (this is the port used by the application) and the service type is NodePort which means it will be reachable externally. +
+ +
+True or False? the target port, in the case of running the following command, will be exposed only on one of the Kubernetes cluster nodes but it will routed to all the pods + +``` +kubectl expose rs some-replicaset --name=replicaset-svc --target-port=2017 --type=NodePort +``` +
+ +False. It will be exposed on every node of the cluster and will be routed to one of the Pods (which belong to the ReplicaSet) +
+ +
+How to verify that a certain service configured to forward the requests to a given pod
+ +Run `kubectl describe service` and see if the IPs from "Endpoints" match any IPs from the output of `kubectl get pod -o wide` +
+ +
+Explain what will happen when running apply on the following block + +``` +apiVersion: v1 +kind: Service +metadata: + name: some-app +spec: + type: NodePort + ports: + - port: 8080 + nodePort: 2017 + protocol: TCP + selector: + type: backend + service: some-app +``` +
+ +It creates a new Service of the type "NodePort" which means it can be used for internal and external communication with the app.
+The port of the application is 8080 and the requests will forwarded to this port. The exposed port is 2017. As a note, this is not a common practice, to specify the nodePort.
+The port used TCP (instead of UDP) and this is also the default so you don't have to specify it.
+The selector used by the Service to know to which Pods to forward the requests. In this case, Pods with the label "type: backend" and "service: some-app".
+
+ +
+How to turn the following service into an external one? + +``` +spec: + selector: + app: some-app + ports: + - protocol: TCP + port: 8081 + targetPort: 8081 +``` +
+ +Adding `type: LoadBalancer` and `nodePort` + +``` +spec: + selector: + app: some-app + type: LoadBalancer + ports: + - protocol: TCP + port: 8081 + targetPort: 8081 + nodePort: 32412 +``` +
+ +
+What would you use to route traffic from outside the Kubernetes cluster to services within a cluster?
+ +Ingress +
+ +
+True or False? When "NodePort" is used, "ClusterIP" will be created automatically?
+ +True +
+ +
+When would you use the "LoadBalancer" type
+ +Mostly when you would like to combine it with cloud provider's load balancer +
+ +
+How would you map a service to an external address?
+ +Using the 'ExternalName' directive. +
+ +
+Describe in detail what happens when you create a service
+ +1. Kubectl sends a request to the API server to create a Service +2. The controller detects there is a new Service +3. Endpoint objects created with the same name as the service, by the controller +4. The controller is using the Service selector to identify the endpoints +5. kube-proxy detects there is a new endpoint object + new service and adds iptables rules to capture traffic to the Service port and redirect it to endpoints +6. kube-dns detects there is a new Service and adds the container record to the dns server +
+ +
+How to list the endpoints of a certain app?
+ +`kubectl get ep ` +
+ +
+How can you find out information on a Service related to a certain Pod if all you can use is kubectl exec --
+ +You can run `kubectl exec -- env` which will give you a couple environment variables related to the Service.
+Variables such as `[SERVICE_NAME]_SERVICE_HOST`, `[SERVICE_NAME]_SERVICE_PORT`, ... +
+ +
+Describe what happens when a container tries to connect with its corresponding Service for the first time. Explain who added each of the components you include in your description
+ + - The container looks at the nameserver defined in /etc/resolv.conf + - The container queries the nameserver so the address is resolved to the Service IP + - Requests sent to the Service IP are forwarded with iptables rules (or other chosen software) to the endpoint(s). + +Explanation as to who added them: + + - The nameserver in the container is added by kubelet during the scheduling of the Pod, by using kube-dns + - The DNS record of the service is added by kube-dns during the Service creation + - iptables rules are added by kube-proxy during Endpoint and Service creation +
+ +
+Describe in high level what happens when you run kubctl expose deployment remo --type=LoadBalancer --port 8080
+ +1. Kubectl sends a request to Kubernetes API to create a Service object +2. Kubernetes asks the cloud provider (e.g. AWS, GCP, Azure) to provision a load balancer +3. The newly created load balancer forwards incoming traffic to relevant worker node(s) which forwards the traffic to the relevant containers +
+ +
+After creating a service that forwards incoming external traffic to the containerized application, how to make sure it works?
+ +You can run `curl :` to examine the output. +
+ +
+An internal load balancer in Kubernetes is called ____ and an external load balancer is called ____
+ +An internal load balancer in Kubernetes is called Service and an external load balancer is Ingress +
+ +### Ingress + +
+What is Ingress?
+ +From Kubernetes docs: "Ingress exposes HTTP and HTTPS routes from outside the cluster to services within the cluster. Traffic routing is controlled by rules defined on the Ingress resource." + +Read more [here](https://kubernetes.io/docs/concepts/services-networking/ingress/) +
+ +
+Complete the following configuration file to make it Ingress + +``` +metadata: + name: someapp-ingress +spec: +``` +
+There are several ways to answer this question. + +``` +apiVersion: networking.k8s.io/v1 +kind: Ingress +metadata: + name: someapp-ingress +spec: + rules: + - host: my.host + http: + paths: + - backend: + serviceName: someapp-internal-service + servicePort: 8080 +``` +
+ + +
+Explain the meaning of "http", "host" and "backend" directives + +``` +apiVersion: networking.k8s.io/v1 +kind: Ingress +metadata: + name: someapp-ingress +spec: + rules: + - host: my.host + http: + paths: + - backend: + serviceName: someapp-internal-service + servicePort: 8080 +``` +
+ +host is the entry point of the cluster so basically a valid domain address that maps to cluster's node IP address
+the http line used for specifying that incoming requests will be forwarded to the internal service using http.
+backend is referencing the internal service (serviceName is the name under metadata and servicePort is the port under the ports section). +
+ +
+Why using a wildcard in ingress host may lead to issues?
+ +The reason you should not wildcard value in a host (like `- host: *`) is because you basically tell your Kubernetes cluster to forward all the traffic to the container where you used this ingress. This may cause the entire cluster to go down. +
+ +
+What is Ingress Controller?
+ +An implementation for Ingress. It's basically another pod (or set of pods) that does evaluates and processes Ingress rules and this it manages all the redirections. + +There are multiple Ingress Controller implementations (the one from Kubernetes is Kubernetes Nginx Ingress Controller). +
+ +
+What are some use cases for using Ingress?
+ +* Multiple sub-domains (multiple host entries, each with its own service) +* One domain with multiple services (multiple paths where each one is mapped to a different service/application) +
+ +
+How to list Ingress in your namespace?
+ +kubectl get ingress +
+ +
+What is Ingress Default Backend?
+ +It specifies what do with an incoming request to the Kubernetes cluster that isn't mapped to any backend (= no rule to for mapping the request to a service). If the default backend service isn't defined, it's recommended to define so users still see some kind of message instead of nothing or unclear error. +
+ +
+How to configure a default backend?
+ +Create Service resource that specifies the name of the default backend as reflected in `kubectl describe ingress ...` and the port under the ports section. +
+ +
+How to configure TLS with Ingress?
+ +Add tls and secretName entries. + +``` +spec: + tls: + - hosts: + - some_app.com + secretName: someapp-secret-tls +``` +
+ +
+True or False? When configuring Ingress with TLS, the Secret component must be in the same namespace as the Ingress component
+ +True +
+ +
+Which Kubernetes concept would you use to control traffic flow at the IP address or port level?
+ +Network Policies +
+ +
+How to scale an application (deplyoment) so it runs more than one instance of the application?
+ +To run two instances of the applicaation? + +`kubectl scale deployment --replicas=2` + +You can specify any other number, given that your application knows how to scale. +
+ +### ReplicaSets + +
+What is the purpose of ReplicaSet?
+ +[kubernetes.io](https://kubernetes.io/docs/concepts/workloads/controllers/replicaset): "A ReplicaSet's purpose is to maintain a stable set of replica Pods running at any given time. As such, it is often used to guarantee the availability of a specified number of identical Pods." + +In simpler words, a ReplicaSet will ensure the specified number of Pods replicas is running for a selected Pod. If there are more Pods than defined in the ReplicaSet, some will be removed. If there are less than what is defined in the ReplicaSet then, then more replicas will be added. +
+ +
+What the following block of lines does? + +``` +spec: + replicas: 2 + selector: + matchLabels: + type: backend + template: + metadata: + labels: + type: backend + spec: + containers: + - name: httpd-yup + image: httpd +``` +
+ +It defines a replicaset for Pods whose type is set to "backend" so at any given point of time there will be 2 concurrent Pods running. +
+ +
+What will happen when a Pod, created by ReplicaSet, is deleted directly with kubectl delete po ...?
+ +The ReplicaSet will create a new Pod in order to reach the desired number of replicas. +
+ +
+True or False? If a ReplicaSet defines 2 replicas but there 3 Pods running matching the ReplicaSet selector, it will do nothing
+ +False. It will terminate one of the Pods to reach the desired state of 2 replicas. +
+ +
+Describe the sequence of events in case of creating a ReplicaSet
+ +* The client (e.g. kubectl) sends a request to the API server to create a ReplicaSet +* The Controller detects there is a new event requesting for a ReplicaSet +* The controller creates new Pod definitions (the exact number depends on what is defined in the ReplicaSet definition) +* The scheduler detects unassigned Pods and decides to which nodes to assign the Pods. This information sent to the API server +* Kubelet detects that two Pods were assigned to the node it's running on (as it constantly watching the API server) +* Kubelet sends requests to the container engine, to create the containers that are part of the Pod +* Kubelet sends a request to the API server to notify it the Pods were created +
+ +
+How to list ReplicaSets in the current namespace?
+ +`kubectl get rs` +
+ +
+Is it possible to delete ReplicaSet without deleting the Pods it created?
+ +Yes, with `--cascase=false`. + +`kubectl delete -f rs.yaml --cascade=false` +
+ +
+What is the default number of replicas if not explicitly specified?
+ +1 +
+ +
+What the following output of kubectl get rs means? + +NAME DESIRED CURRENT READY AGE +web 2 2 0 2m23s +
+ +The replicaset `web` has 2 replicas. It seems that the containers inside the Pod(s) are not yet running since the value of READY is 0. It might be normal since it takes time for some containers to start running and it might be due to an error. Running `kubectl describe po POD_NAME` or `kubectl logs POD_NAME` can give us more information. +
+ +
+True or False? Pods specified by the selector field of ReplicaSet must be created by the ReplicaSet itself
+ +False. The Pods can be already running and initially they can be created by any object. It doesn't matter for the ReplicaSet and not a requirement for it to acquire and monitor them. +
+ +
+True or False? In case of a ReplicaSet, if Pods specified in the selector field don't exists, the ReplicaSet will wait for them to run before doing anything
+ +False. It will take care of running the missing Pods. +
+ +
+In case of a ReplicaSet, Which field is mandatory in the spec section?
+ +The field `template` in spec section is mandatory. It's used by the ReplicaSet to create new Pods when needed. +
+ +
+You've created a ReplicaSet, how to check whether the ReplicaSet found matching Pods or it created new Pods?
+ +`kubectl describe rs ` + +It will be visible under `Events` (the very last lines) +
+ +
+True or False? Deleting a ReplicaSet will delete the Pods it created
+ +True (and not only the Pods but anything else it created). +
+ +
+True or False? Removing the label from a Pod that is tracked by a ReplicaSet, will cause the ReplicaSet to create a new Pod
+ +True. When the label, used by a ReplicaSet in the selector field, removed from a Pod, that Pod no longer controlled by the ReplicaSet and the ReplicaSet will create a new Pod to compensate for the one it "lost". +
+ +
+How to scale a deployment to 8 replicas?
+ +kubectl scale deploy --replicas=8 +
+ +
+ReplicaSets are running the moment the user executed the command to create them (like kubectl create -f rs.yaml)
+ +False. It can take some time, depends on what exactly you are running. To see if they are up and running, run `kubectl get rs` and watch the 'READY' column. +
+ +
+How to expose a ReplicaSet as a new service?
+ +`kubectl expose rs --name= --target-port= --type=NodePort` + +Few notes: + - the target port depends on which port the app is using in the container + - type can be different and doesn't has to be specifically "NodePort" +
+ +
+Fix the following ReplicaSet definition + +```yaml +apiVersion: apps/v1 +kind: ReplicaCet +metadata: + name: redis + labels: + app: redis + tier: cache +spec: + selector: + matchLabels: + tier: cache + template: + metadata: + labels: + tier: cachy + spec: + containers: + - name: redis + image: redis +``` +
+ +kind should be ReplicaSet and not ReplicaCet :) +
+ +
+Fix the following ReplicaSet definition + +```yaml +apiVersion: apps/v1 +kind: ReplicaSet +metadata: + name: redis + labels: + app: redis + tier: cache +spec: + selector: + matchLabels: + tier: cache + template: + metadata: + labels: + tier: cachy + spec: + containers: + - name: redis + image: redis +``` +
+ +The selector doesn't match the label (cache vs cachy). To solve it, fix cachy so it's cache instead. +
+ +
+How to check which container image was used as part of replica set called "repli"?
+ +`k describe rs repli | grep -i image` +
+ +
+How to check how many Pods are ready as part of a replica set called "repli"?
+ +`k describe rs repli | grep -i "Pods Status"` +
+ +
+How to delete a replica set called "rori"?
+ +`k delete rs rori` +
+ +
+How to modify a replica set called "rori" to use a different image?
+ +`k edis rs rori` +
+ +
+Scale up a replica set called "rori" to run 5 Pods instead of 2
+ +`k scale rs rori --replicas=5` +
+ +
+Scale down a replica set called "rori" to run 1 Pod instead of 5
+ +`k scale rs rori --replicas=1` +
+ +### DaemonSet + +
+What's a DaemonSet?
+ +[Kubernetes.io](https://kubernetes.io/docs/concepts/workloads/controllers/daemonset): "A DaemonSet ensures that all (or some) Nodes run a copy of a Pod. As nodes are added to the cluster, Pods are added to them. As nodes are removed from the cluster, those Pods are garbage collected. Deleting a DaemonSet will clean up the Pods it created." +
+ +
+What's the difference between a ReplicaSet and DaemonSet?
+ +A ReplicaSet's purpose is to maintain a stable set of replica Pods running at any given time. +A DaemonSet ensures that all Nodes run a copy of a Pod. +
+ +
+What are some use cases for using a DaemonSet?
+ +* Monitoring: You would like to perform monitoring on every node part of cluster. For example datadog pod runs on every node using a daemonset +* Logging: You would like to having logging set up on every node part of your cluster +* Networking: there is networking component you need on every node for all nodes to communicate between them +
+ +
+How DaemonSet works?
+ +Historically, up 1.12, it was done with NodeName attribute. + +Starting 1.12, it's achieved with regular scheduler and node affinity. +
+ +#### DaemonSet - Commands + +
+How to list all daemonsets in the current namespace?
+ +`kubectl get ds` +
+ +### StatefulSet + +
+Explain StatefulSet
+ +StatefulSet is the workload API object used to manage stateful applications. Manages the deployment and scaling of a set of Pods, and provides guarantees about the ordering and uniqueness of these Pods.[Learn more](https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/) +
+ +### Storage + +#### Volumes + +
+What is a volume in regards to Kubernetes?
+ +A directory accessible by the containers inside a certain Pod and containers. The mechanism responsible for creating the directory, managing it, ... mainly depends on the volume type. + +
+ +
+What volume types are you familiar with?
+ +* emptyDir: created when a Pod assigned to a node and ceases to exist when the Pod is no longer running on that node +* hostPath: mounts a path from the host itself. Usually not used due to security risks but has multiple use-cases where it's needed like access to some internal host paths (`/sys`, `/var/lib`, etc.) + +
+ +
+Which problems, volumes in Kubernetes solve?
+ +1. Sharing files between containers running in the same Pod +2. Storage in containers is ephemeral - it usually doesn't last for long. For example, when a container crashes, you lose all on-disk data. Certain volumes allows to manage such situation by persistent volumes + +
+ +
+Explain ephemeral volume types vs. persistent volumes in regards to Pods
+ +Ephemeral volume types have the lifetime of a pod as opposed to persistent volumes which exist beyond the lifetime of a Pod. + +
+ +
+Provide at least one use-case for each of the following volume types: + +* emptyDir +* hostPath +
+ +* EmptyDir: You need a temporary data that you can afford to lose if the Pod is deleted. For example short-lived data required for one-time operations. +* hostPath: You need access to paths on the host itself (like data from `/sys` or data generated in `/var/lib`) +
+ +### Networking + +
+True or False? By default there is no communication between two Pods in two different namespaces
+ +False. By default two Pods in two different namespaces are able to communicate with each other. + +Try it for yourself: + +kubectl run test-prod -n prod --image ubuntu -- sleep 2000000000 +kubectl run test-dev -n dev --image ubuntu -- sleep 2000000000 + +`k describe po test-prod -n prod` to get the IP of test-prod Pod. + +Access dev Pod: `kubectl exec --stdin --tty test-dev -n dev -- /bin/bash` + +And ping the IP of test-prod Pod you get earlier.You'll see that there is communication between the two pods, in two separate namespaces. + +
+ +### Network Policies + +
+Explain Network Policies
+ +[kubernetes.io](https://kubernetes.io/docs/concepts/services-networking/network-policies): "NetworkPolicies are an application-centric construct which allow you to specify how a pod is allowed to communicate with various network "entities"..." + +In simpler words, Network Policies specify how pods are allowed/disallowed to communicate with each other and/or other network endpoints. + +
+ +
+What are some use cases for using Network Policies?
+ + - Security: You want to prevent from everyone to communicate with a certain pod for security reasons + - Controlling network traffic: You would like to deny network flow between two specific nodes + +
+ +
+True or False? If no network policies are applied to a pod, then no connections to or from it are allowed
+ +False. By default pods are non-isolated. +
+ +
+In case of two pods, if there is an egress policy on the source denining traffic and ingress policy on the destination that allows traffic then, traffic will be allowed or denied?
+ +Denied. Both source and destination policies has to allow traffic for it to be allowed. +
+ +
+Where Kubernetes cluster stores the cluster state?
+ +etcd +
+ +### etcd + +
+What is etcd?
+ +etcd is an open source distributed key-value store used to hold and manage the critical information that distributed systems need to keep running. + +[Read more here](https://www.redhat.com/en/topics/containers/what-is-etcd) + +
+ +
+True or False? Etcd holds the current status of any kubernetes component
+ +True +
+ +
+True or False? The API server is the only component which communicates directly with etcd
+ +True +
+ +
+True or False? application data is not stored in etcd
+ +True +
+ +
+Why etcd? Why not some SQL or NoSQL database?
+ +When chosen as the data store etcd was (and still is of course): + +* Highly Available - you can deploy multiple nodes +* Fully Replicated - any node in etcd cluster is "primary" node and has full access to the data +* Consistent - reads return latest data +* Secured - supports both TLS and SSL +* Speed - high performance data store (10k writes per sec!) +
+ +### Namespaces + +
+What are namespaces?
+ +Namespaces allow you split your cluster into virtual clusters where you can group your applications in a way that makes sense and is completely separated from the other groups (so you can for example create an app with the same name in two different namespaces) +
+ +
+Why to use namespaces? What is the problem with using one default namespace?
+ +When using the default namespace alone, it becomes hard over time to get an overview of all the applications you manage in your cluster. Namespaces make it easier to organize the applications into groups that makes sense, like a namespace of all the monitoring applications and a namespace for all the security applications, etc. + +Namespaces can also be useful for managing Blue/Green environments where each namespace can include a different version of an app and also share resources that are in other namespaces (namespaces like logging, monitoring, etc.). + +Another use case for namespaces is one cluster, multiple teams. When multiple teams use the same cluster, they might end up stepping on each others toes. For example if they end up creating an app with the same name it means one of the teams overridden the app of the other team because there can't be too apps in Kubernetes with the same name (in the same namespace). +
+ +
+True or False? When a namespace is deleted all resources in that namespace are not deleted but moved to another default namespace
+ +False. When a namespace is deleted, the resources in that namespace are deleted as well. +
+ +
+What special namespaces are there by default when creating a Kubernetes cluster?
+ +* default +* kube-system +* kube-public +* kube-node-lease +
+ +
+What can you find in kube-system namespace?
+ +* Master and Kubectl processes +* System processes +
+ +
+While namespaces do provide scope for resources, they are not isolating them
+ +True. Try create two pods in two separate namespaces for example, and you'll see there is a connection between the two. +
+ +#### Namespaces - commands + +
+How to list all namespaces?
+ +`kubectl get namespaces` OR `kubectl get ns` + +
+ +
+Create a namespace called 'alle'
+ +`k create ns alle` + +
+ +
+Check how many namespaces are there
+ +`k get ns --no-headers | wc -l` + +
+ +
+Check how many pods exist in the "dev" namespace
+ +`k get po -n dev` + +
+ +
+Create a pod called "kartos" in the namespace dev. The pod should be using the "redis" image.
+ +If the namespace doesn't exist already: `k create ns dev` + +`k run kratos --image=redis -n dev` + +
+ +
+You are looking for a Pod called "atreus". How to check in which namespace it runs?
+ +`k get po -A | grep atreus` + +
+ +
+What kube-public contains?
+ +* A configmap, which contains cluster information +* Publicly accessible data +
+ +
+How to get the name of the current namespace?
+ +`kubectl config view | grep namespace` +
+ +
+What kube-node-lease contains?
+ +It holds information on heartbeats of nodes. Each node gets an object which holds information about its availability. +
+ +
+True or False? With namespaces you can limit the resources consumed by the users/teams
+ +True. With namespaces you can limit CPU, RAM and storage usage. +
+ +
+How to switch to another namespace? In other words how to change active namespace?
+ +`kubectl config set-context --current --namespace=some-namespace` and validate with `kubectl config view --minify | grep namespace:` + +OR + +`kubens some-namespace` +
+ +#### Resources Quota + +
+What is Resource Quota?
+ +Resource quota provides constraints that limit aggregate resource consumption per namespace. It can limit the quantity of objects that can be created in a namespace by type, as well as the total amount of compute resources that may be consumed by resources in that namespace. +
+ +
+How to create a Resource Quota?
+ +kubectl create quota some-quota --hard-cpu=2,pods=2 +
+ +
+Which resources are accessible from different namespaces?
+ +Services. +
+ +
+Which service and in which namespace the following file is referencing? + +``` +apiVersion: v1 +kind: ConfigMap +metadata: + name: some-configmap +data: + some_url: samurai.jack +``` +
+ +It's referencing the service "samurai" in the namespace called "jack". +
+ +
+Which components can't be created within a namespace?
+ +Volume and Node. +
+ +
+How to list all the components that bound to a namespace?
+ +`kubectl api-resources --namespaced=true` +
+ +
+How to create components in a namespace?
+ +One way is by specifying --namespace like this: `kubectl apply -f my_component.yaml --namespace=some-namespace` +Another way is by specifying it in the YAML itself: + +``` +apiVersion: v1 +kind: ConfigMap +metadata: + name: some-configmap + namespace: some-namespace +``` + +and you can verify with: `kubectl get configmap -n some-namespace` +
+ +
+How to execute the command "ls" in an existing pod?
+ +kubectl exec some-pod -it -- ls +
+ +
+How to create a service that exposes a deployment?
+ +kubectl expose deploy some-deployment --port=80 --target-port=8080 +
+ +
+How to create a pod and a service with one command?
+ +kubectl run nginx --image=nginx --restart=Never --port 80 --expose +
+ +
+Describe in detail what the following command does kubectl create deployment kubernetes-httpd --image=httpd
+
+ +
+Why to create kind deployment, if pods can be launched with replicaset?
+
+ +
+How to get list of resources which are not bound to a specific namespace?
+ +kubectl api-resources --namespaced=false +
+ +
+How to delete all pods whose status is not "Running"?
+ +kubectl delete pods --field-selector=status.phase!='Running' +
+ +
+How to display the resources usages of pods?
+ +kubectl top pod +
+ +
+Perhaps a general question but, you suspect one of the pods is having issues, you don't know what exactly. What do you do?
+ +Start by inspecting the pods status. we can use the command `kubectl get pods` (--all-namespaces for pods in system namespace)
+ +If we see "Error" status, we can keep debugging by running the command `kubectl describe pod [name]`. In case we still don't see anything useful we can try stern for log tailing.
+ +In case we find out there was a temporary issue with the pod or the system, we can try restarting the pod with the following `kubectl scale deployment [name] --replicas=0`
+ +Setting the replicas to 0 will shut down the process. Now start it with `kubectl scale deployment [name] --replicas=1` +
+ +
+What happens what pods are using too much memory? (more than its limit)
+ +They become candidates to for termination. +
+ +
+Describe how roll-back works
+
+ +
+True or False? Memory is a compressible resource, meaning that when a container reach the memory limit, it will keep running
+ +False. CPU is a compressible resource while memory is a non compressible resource - once a container reached the memory limit, it will be terminated. +
+ +### Operators + +
+What is an Operator?
+ +Explained [here](https://kubernetes.io/docs/concepts/extend-kubernetes/operator) + +"Operators are software extensions to Kubernetes that make use of custom resources to manage applications and their components. Operators follow Kubernetes principles, notably the control loop." + +In simpler words, you can think about an operator as a custom control loop in Kubernetes. +
+ +
+Why do we need Operators?
+ +The process of managing stateful applications in Kubernetes isn't as straightforward as managing stateless applications where reaching the desired status and upgrades are both handled the same way for every replica. In stateful applications, upgrading each replica might require different handling due to the stateful nature of the app, each replica might be in a different status. As a result, we often need a human operator to manage stateful applications. Kubernetes Operator is suppose to assist with this. + +This also help with automating a standard process on multiple Kubernetes clusters +
+ +
+What components the Operator consists of?
+ +1. CRD (Custom Resource Definition) - You are fanmiliar with Kubernetes resources like Deployment, Pod, Service, etc. CRD is also a resource, but one that you or the developer the operator defines. +2. Controller - Custom control loop which runs against the CRD + +
+ +
+Explain CRD
+ +CRD is Custom Resource Definitions. It's custom Kubernetes component which extends K8s API. + +TODO(abregman): add more info. + +
+ +
+How Operator works?
+ +It uses the control loop used by Kubernetes in general. It watches for changes in the application state. The difference is that is uses a custom control loop. + +In addition, it also makes use of CRD's (Custom Resources Definitions) so basically it extends Kubernetes API. + +
+ +
+True or False? Kubernetes Operator used for stateful applications
+ +True +
+ +
+Explain what is the OLM (Operator Lifecycle Manager) and what is it used for
+ +
+ +
+What is the Operator Framework?
+ +open source toolkit used to manage k8s native applications, called operators, in an automated and efficient way. + +
+ +
+What components the Operator Framework consists of?
+ +1. Operator SDK - allows developers to build operators +2. Operator Lifecycle Manager - helps to install, update and generally manage the lifecycle of all operators +3. Operator Metering - Enables usage reporting for operators that provide specialized services +4. +
+ +
+Describe in detail what is the Operator Lifecycle Manager
+ +It's part of the Operator Framework, used for managing the lifecycle of operators. It basically extends Kubernetes so a user can use a declarative way to manage operators (installation, upgrade, ...). + +
+ +
+What openshift-operator-lifecycle-manager namespace includes?
+ +It includes: + + * catalog-operator - Resolving and installing ClusterServiceVersions the resource they specify. + * olm-operator - Deploys applications defined by ClusterServiceVersion resource + +
+ +
+What is kubconfig? What do you use it for?
+ +A kubeconfig file is a file used to configure access to Kubernetes when used in conjunction with the kubectl commandline tool (or other clients). +Use kubeconfig files to organize information about clusters, users, namespaces, and authentication mechanisms. +
+ +
+Would you use Helm, Go or something else for creating an Operator?
+ +Depends on the scope and maturity of the Operator. If it mainly covers installation and upgrades, Helm might be enough. If you want to go for Lifecycle management, insights and auto-pilot, this is where you'd probably use Go. +
+ +
+Are there any tools, projects you are using for building Operators?
+ +This one is based more on a personal experience and taste... + +* Operator Framework +* Kubebuilder +* Controller Runtime +... +
+ +### Secrets + +
+Explain Kubernetes Secrets
+ +Secrets let you store and manage sensitive information (passwords, ssh keys, etc.) + +
+ +
+How to create a Secret from a key and value?
+ +`kubectl create secret generic some-secret --from-literal=password='donttellmypassword'` + +
+ +
+How to create a Secret from a file?
+ +`kubectl create secret generic some-secret --from-file=/some/file.txt` +
+ +
+What type: Opaque in a secret file means? What other types are there?
+ +Opaque is the default type used for key-value pairs. +
+ +
+True or False? storing data in a Secret component makes it automatically secured
+ +False. Some known security mechanisms like "encryption" aren't enabled by default. +
+ +
+What is the problem with the following Secret file: + +``` +apiVersion: v1 +kind: Secret +metadata: + name: some-secret +type: Opaque +data: + password: mySecretPassword +``` +
+ +Password isn't encrypted. +You should run something like this: `echo -n 'mySecretPassword' | base64` and paste the result to the file instead of using plain-text. + +
+ +
+What the following in a Deployment configuration file means? + +``` +spec: + containers: + - name: USER_PASSWORD + valueFrom: + secretKeyRef: + name: some-secret + key: password +``` +
+ +USER_PASSWORD environment variable will store the value from password key in the secret called "some-secret" +In other words, you reference a value from a Kubernetes Secret. + +
+ +
+How to commit secrets to Git and in general how to use encrypted secrets?
+ +One possible process would be as follows: + +1. You create a Kubernetes secret (but don't commit it) +2. You encrypt it using some 3rd party project (.e.g kubeseal) +3. You apply the seald/encrypted secret +4. You commit the the sealed secret to Git +5. You deploy an application that requires the secret and it can be automatically decrypted by using for example a Bitnami Sealed secrets controller +
+ +### Volumes + +
+True or False? Kubernetes provides data persistence out of the box, so when you restart a pod, data is saved
+ +False +
+ +
+Explain "Persistent Volumes". Why do we need it?
+ +Persistent Volumes allow us to save data so basically they provide storage that doesn't depend on the pod lifecycle. +
+ +
+True or False? Persistent Volume must be available to all nodes because the pod can restart on any of them
+ +True +
+ +
+What types of persistent volumes are there?
+ +* NFS +* iSCSI +* CephFS +* ... +
+ +
+What is PersistentVolumeClaim?
+
+ +
+Explain Volume Snapshots
+ +Volume snapshots let you create a copy of your volume at a specific point in time. +
+ +
+True or False? Kubernetes manages data persistence
+ +False +
+ +
+Explain Storage Classes
+
+ +
+Explain "Dynamic Provisioning" and "Static Provisioning"
+ +The main difference relies on the moment when you want to configure storage. For instance, if you need to pre-populate data in a volume, you choose static provisioning. Whereas, if you need to create volumes on demand, you go for dynamic provisioning. +
+ +
+Explain Access Modes
+
+ +
+What is CSI Volume Cloning?
+
+ +
+Explain "Ephemeral Volumes"
+
+ +
+What types of ephemeral volumes Kubernetes supports?
+
+ +
+What is Reclaim Policy?
+
+ +
+What reclaim policies are there?
+ +* Retain +* Recycle +* Delete +
+ +### Access Control + +
+What is RBAC?
+ +RBAC in Kubernetes is the mechanism that enables you to configure fine-grained and specific sets of permissions that define how a given user, or group of users, can interact with any Kubernetes object in cluster, or in a specific Namespace of cluster. +
+ +
+Explain the Role and RoleBinding" objects
+
+ +
+What is the difference between Role and ClusterRole objects?
+ +The difference between them is that a Role is used at a namespace level whereas a ClusterRole is for the entire cluster. +
+ +
+Explain what are "Service Accounts" and in which scenario would use create/use one
+ +[Kubernetes.io](https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account): "A service account provides an identity for processes that run in a Pod." + +An example of when to use one: +You define a pipeline that needs to build and push an image. In order to have sufficient permissions to build an push an image, that pipeline would require a service account with sufficient permissions. +
+ +
+What happens you create a pod and you DON'T specify a service account?
+ +The pod is automatically assigned with the default service account (in the namespace where the pod is running). +
+ +
+Explain how Service Accounts are different from User Accounts
+ + - User accounts are global while Service accounts unique per namespace + - User accounts are meant for humans or client processes while Service accounts are for processes which run in pods +
+ +
+How to list Service Accounts?
+ +`kubectl get serviceaccounts` +
+ +
+Explain "Security Context"
+ +[kubernetes.io](https://kubernetes.io/docs/tasks/configure-pod-container/security-context): "A security context defines privilege and access control settings for a Pod or Container." +
+ +### Patterns + + + +### CronJob + +
+Explain what is CronJob and what is it used for
+ +A CronJob creates Jobs on a repeating schedule. One CronJob object is like one line of a crontab (cron table) file. It runs a job periodically on a given schedule, written in Cron format. +
+ +
+What possible issue can arise from using the following spec and how to fix it? + +``` +apiVersion: batch/v1beta1 +kind: CronJob +metadata: + name: some-cron-job +spec: + schedule: '*/1 * * * *' + startingDeadlineSeconds: 10 + concurrencyPolicy: Allow +``` +
+ +If the cron job fails, the next job will not replace the previous one due to the "concurrencyPolicy" value which is "Allow". It will keep spawning new jobs and so eventually the system will be filled with failed cron jobs. +To avoid such problem, the "concurrencyPolicy" value should be either "Replace" or "Forbid". +
+ +
+What issue might arise from using the following CronJob and how to fix it? + +``` +apiVersion: batch/v1beta1 +kind: CronJob +metadata: + name: "some-cron-job" +spec: + schedule: '*/1 * * * *' +jobTemplate: + spec: + template: + spec: + restartPolicy: Never + concurrencyPolicy: Forbid + successfulJobsHistoryLimit: 1 + failedJobsHistoryLimit: 1 +``` +
+ +The following lines placed under the template: + +``` +concurrencyPolicy: Forbid +successfulJobsHistoryLimit: 1 +failedJobsHistoryLimit: 1 +``` + +As a result this configuration isn't part of the cron job spec hence the cron job has no limits which can cause issues like OOM and potentially lead to API server being down.
+To fix it, these lines should placed in the spec of the cron job, above or under the "schedule" directive in the above example. +
+ +### Misc + +
+Explain Imperative Management vs. Declarative Management
+ +
+ +
+Explain what Kubernetes Service Discovery means
+
+ +
+You have one Kubernetes cluster and multiple teams that would like to use it. You would like to limit the resources each team consumes in the cluster. Which Kubernetes concept would you use for that?
+ +Namespaces will allow to limit resources and also make sure there are no collisions between teams when working in the cluster (like creating an app with the same name). +
+ +
+What Kube Proxy does?
+ Kube Proxy is a network proxy that runs on each node in your cluster, implementing part of the Kubernetes Service concept +
+ +
+What "Resources Quotas" are used for and how?
+
+ +
+Explain ConfigMap
+ +Separate configuration from pods. +It's good for cases where you might need to change configuration at some point but you don't want to restart the application or rebuild the image so you create a ConfigMap and connect it to a pod but externally to the pod. + +Overall it's good for: +* Sharing the same configuration between different pods +* Storing external to the pod configuration +
+ +
+How to use ConfigMaps?
+ +1. Create it (from key&value, a file or an env file) +2. Attach it. Mount a configmap as a volume +
+ +
+True or False? Sensitive data, like credentials, should be stored in a ConfigMap
+ +False. Use secret. +
+ +
+Explain "Horizontal Pod Autoscaler"
+ +In Kubernetes, a HorizontalPodAutoscaler automatically updates a workload resource with the aim of automatically scaling the workload to match demand. +
+ +
+When you delete a pod, is it deleted instantly? (a moment after running the command)
+
+ +
+What does being cloud-native mean?
+ The term cloud native refers to the concept of building and running applications to take advantage of the distributed computing offered by the cloud delivery model. +
+ +
+Explain the pet and cattle approach of infrastructure with respect to kubernetes
+
+ +
+Describe how you one proceeds to run a containerized web app in K8s, which should be reachable from a public URL.
+
+ +
+How would you troubleshoot your cluster if some applications are not reachable any more?
+
+ +
+Describe what CustomResourceDefinitions there are in the Kubernetes world? What they can be used for?
+
+ +
+ How does scheduling work in kubernetes?
+ +The control plane component kube-scheduler asks the following questions, +1. What to schedule? It tries to understand the pod-definition specifications +2. Which node to schedule? It tries to determine the best node with available resources to spin a pod +3. Binds the Pod to a given node + +View more [here](https://www.youtube.com/watch?v=rDCWxkvPlAw) +
+ +
+ How are labels and selectors used?
+
+ +
+What QoS classes are there?
+ +* Guaranteed +* Burstable +* BestEffort +
+ +
+Explain Labels. What are they and why would one use them?
+ +Kubernetes labels are key-value pairs that can connect identifying metadata with Kubernetes objects. +
+ +
+Explain Selectors
+
+ +
+What is Kubeconfig?
+
+ +### Gatekeeper + +
+What is Gatekeeper?
+ +[Gatekeeper docs](https://open-policy-agent.github.io/gatekeeper/website/docs): "Gatekeeper is a validating (mutating TBA) webhook that enforces CRD-based policies executed by Open Policy Agent" +
+ +
+Explain how Gatekeeper works
+ +On every request sent to the Kubernetes cluster, Gatekeeper sends the policies and the resources to OPA (Open Policy Agent) to check if it violates any policy. If it does, Gatekeeper will return the policy error message back. If it isn't violates any policy, the request will reach the cluster. +
+ +### Policy Testing + +
+What is Conftest?
+ +Conftest allows you to write tests against structured files. You can think of it as tests library for Kubernetes resources.
+It is mostly used in testing environments such as CI pipelines or local hooks. +
+ +
+What is Datree? How is it different from Conftest?
+ +Same as Conftest, it is used for policy testing and enforcement. The difference is that it comes with built-in policies. +
+ +### Helm + +
+What is Helm?
+ +Package manager for Kubernetes. Basically the ability to package YAML files and distribute them to other users and apply them in the cluster(s). + +As a concept it's quite common and can be found in many platforms and services. Think for example on package managers in operating systems. If you use Fedora/RHEL that would be dnf. If you use Ubuntu then, apt. If you don't use Linux, then a different question should be asked and it's why? but that's another topic :) +
+ +
+Why do we need Helm? What would be the use case for using it?
+ +Sometimes when you would like to deploy a certain application to your cluster, you need to create multiple YAML files/components like: Secret, Service, ConfigMap, etc. This can be tedious task. So it would make sense to ease the process by introducing something that will allow us to share these bundle of YAMLs every time we would like to add an application to our cluster. This something is called Helm. + +A common scenario is having multiple Kubernetes clusters (prod, dev, staging). Instead of individually applying different YAMLs in each cluster, it makes more sense to create one Chart and install it in every cluster. + +Another scenario is, you would like to share what you've created with the community. For people and companies to easily deploy your application in their cluster. +
+ +
+Explain "Helm Charts"
+ +Helm Charts is a bundle of YAML files. A bundle that you can consume from repositories or create your own and publish it to the repositories. +
+ +
+It is said that Helm is also Templating Engine. What does it mean?
+ +It is useful for scenarios where you have multiple applications and all are similar, so there are minor differences in their configuration files and most values are the same. With Helm you can define a common blueprint for all of them and the values that are not fixed and change can be placeholders. This is called a template file and it looks similar to the following + +``` +apiVersion: v1 +kind: Pod +metadata: + name: {[ .Values.name ]} +spec: + containers: + - name: {{ .Values.container.name }} + image: {{ .Values.container.image }} + port: {{ .Values.container.port }} +``` + +The values themselves will in separate file: + +``` +name: some-app +container: + name: some-app-container + image: some-app-image + port: 1991 +``` +
+ +
+What are some use cases for using Helm template file?
+ +* Deploy the same application across multiple different environments +* CI/CD +
+ +
+Explain the Helm Chart Directory Structure
+ +someChart/ -> the name of the chart + Chart.yaml -> meta information on the chart + values.yaml -> values for template files + charts/ -> chart dependencies + templates/ -> templates files :) +
+ +
+How Helm supports release management?
+ +Helm allows you to upgrade, remove and rollback to previous versions of charts. In version 2 of Helm it was with what is known as "Tiller". In version 3, it was removed due to security concerns. +
+ +#### Commands + +
+How do you search for charts?
+ +`helm search hub [some_keyword]` +
+ +
+Is it possible to override values in values.yaml file when installing a chart?
+Yes. You can pass another values file: +`helm install --values=override-values.yaml [CHART_NAME]` + +Or directly on the command line: `helm install --set some_key=some_value` +
+ +
+How do you list deployed releases?
+ +`helm ls` or `helm list` +
+ +
+How to execute a rollback?
+ +`helm rollback RELEASE_NAME REVISION_ID` +
+ +
+How to view revision history for a certain release?
+ +`helm history RELEASE_NAME` +
+ +
+How to upgrade a release?
+ +`helm upgrade RELEASE_NAME CHART_NAME` +
+ +### Security + +
+What security best practices do you follow in regards to the Kubernetes cluster?
+ + * Secure inter-service communication (one way is to use Istio to provide mutual TLS) + * Isolate different resources into separate namespaces based on some logical groups + * Use supported container runtime (if you use Docker then drop it because it's deprecated. You might want to CRI-O as an engine and podman for CLI) + * Test properly changes to the cluster (e.g. consider using Datree to prevent kubernetes misconfigurations) + * Limit who can do what (by using for example OPA gatekeeper) in the cluster + * Use NetworkPolicy to apply network security + * Consider using tools (e.g. Falco) for monitoring threats +
+ +### Troubleshooting Scenarios + +
+Running kubectl get pods you see Pods in "Pending" status. What would you do?
+ +One possible path is to run `kubectl describe pod ` to get more details.
+You might see one of the following: + * Cluster is full. In this case, extend the cluster. + * ResourcesQuota limits are met. In this case you might want to modify them + * Check if PersistentVolumeClaim mount is pending + +If none of the above helped, run the command (`get pods`) with `-o wide` to see if the node is assigned to a node. If not, there might be an issue with scheduler. +
+ +
+Users unable to reach an application running on a Pod on Kubernetes. What might be the issue and how to check?
+ +One possible path is to start with checking the Pod status. +1. Is the Pod pending? if yes, check for the reason with `kubectl describe pod ` +TODO: finish this... +
+ +### Istio + +
+What is Istio? What is it used for?
+ +Istio is an open source service mesh that helps organizations run distributed, microservices-based apps anywhere. Istio enables organizations to secure, connect, and monitor microservices, so they can modernize their enterprise apps more swiftly and securely. +
+ +### Controllers + +
+What are controllers?
+ +[Kubernetes.io](https://kubernetes.io/docs/concepts/architecture/controller): "In Kubernetes, controllers are control loops that watch the state of your cluster, then make or request changes where needed. Each controller tries to move the current cluster state closer to the desired state." +
+ +
+Name two controllers you are familiar with
+ +1. Node Contorller: manages the nodes of a cluster. Among other things, the controller is responsible for monitoring nodes' health - if the node is suddenly unreachable it will evacuate all the pods running on it and will mark the node status accordingly. +2. Replication Controller - monitors the status of pod replicas based on what should be running. It makes sure the number of pods that should be running is actually running +
+ +
+What process is responsible for running and installing the different controllers?
+ +Kube-Controller-Manager +
+ +
+What is the control loop? How it works?
+ +Explained [here](https://www.youtube.com/watch?v=i9V4oCa5f9I) +
+ +
+What are all the phases/steps of a control loop?
+ +- Observe - identify the cluster current state +- Diff - Identify whether a diff exists between current state and desired state +- Act - Bring current cluster state to the desired state (basically reach a state where there is no diff) +
+ +### Scheduler + +
+True of False? The scheduler is responsible for both deciding where a Pod will run and actually running it
+ +False. While the scheduler is responsible for choosing the node on which the Pod will run, Kubelet is the one that actually runs the Pod. +
+ +
+How to schedule a pod on a node called "node1"?
+ +`k run some-pod --image=redix -o yaml --dry-run=client > pod.yaml` + +`vi pod.yaml` and add: + +``` +spec: + nodeName: node1 +``` + +`k apply -f pod.yaml` + +Note: if you don't have a node1 in your cluster the Pod will be stuck on "Pending" state. +
+ +#### Node Affinity + +
+Using node affinity, set a Pod to schedule on a node where the key is "region" and value is either "asia" or "emea"
+ +`vi pod.yaml` + +```yaml +affinity: + nodeAffinity: + requiredDuringSchedlingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: region + operator: In + values: + - asia + - emea +``` +
+ +
+Using node affinity, set a Pod to never schedule on a node where the key is "region" and value is "neverland"
+ +`vi pod.yaml` + +```yaml +affinity: + nodeAffinity: + requiredDuringSchedlingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: region + operator: NotIn + values: + - neverland +``` +
+ +
+True of False? Using the node affinity type "requiredDuringSchedlingIgnoredDuringExecution" means the scheduler can't schedule unless the rule is met
+ +True +
+ +
+True of False? Using the node affinity type "preferredDuringSchedlingIgnoredDuringExecution" means the scheduler can't schedule unless the rule is met
+ +False. The scheduler tries to find a node that meets the requirements/rules and if it doesn't it will schedule the Pod anyway. +
+ +
+Can you deploy multiple schedulers?
+ +Yes, it is possible. You can run another pod with a command similar to: + +``` +spec: + containers: + - command: + - kube-scheduler + - --address=127.0.0.1 + - --leader-elect=true + - --scheduler-name=some-custom-scheduler +... +``` +
+ +
+Assuming you have multiple schedulers, how to know which scheduler was used for a given Pod?
+ +Running `kubectl get events` you can see which scheduler was used. +
+ +
+You want to run a new Pod and you would like it to be scheduled by a custom scheduler. How to achieve it?
+ +Add the following to the spec of the Pod: + +``` +spec: + schedulerName: some-custom-scheduler +``` +
+ +### Taints + +
+Check if there are taints on node "master"
+ +`k describe no master | grep -i taints` +
+ +
+Create a taint on one of the nodes in your cluster with key of "app" and value of "web" and effect of "NoSchedule". Verify it was applied
+ +`k taint node minikube app=web:NoSchedule` + +`k describe no minikube | grep -i taints` +
+ +
+You applied a taint with k taint node minikube app=web:NoSchedule on the only node in your cluster and then executed kubectl run some-pod --image=redis. What will happen?
+ +The Pod will remain in "Pending" status due to the only node in the cluster having a taint of "app=web". +
+ +
+You applied a taint with k taint node minikube app=web:NoSchedule on the only node in your cluster and then executed kubectl run some-pod --image=redis but the Pod is in pending state. How to fix it?
+ +`kubectl edit po some-pod` and add the following + +``` + - effect: NoSchedule + key: app + operator: Equal + value: web +``` + +Exit and save. The pod should be in Running state now. +
+ +
+Remove an existing taint from one of the nodes in your cluster
+ +`k taint node minikube app=web:NoSchedule-` +
+ +
+What taint effects are there? Explain each one of them
+ +`NoSchedule`: prevents from resources to be scheduled on a certain node +`PreferNoSchedule`: will prefer to shcedule resources on other nodes before resorting to scheduling the resource on the chosen node (on which the taint was applied) +`NoExecute`: Applying "NoSchedule" will not evict already running Pods (or other resources) from the node as opposed to "NoExecute" which will evict any already running resource from the Node +
+ +### Resource Limits + +
+Explain why one would specify resource limits in regards to Pods
+ +* You know how much RAM and/or CPU your app should be consuming and anything above that is not valid +* You would like to make sure that everyone can run their apps in the cluster and resources are not being solely used by one type of application +
+ +
+True or False? Resource limits applied on a Pod level meaning, if limits is 2gb RAM and there are two container in a Pod that it's 1gb RAM each
+ +False. It's per container and not per Pod. +
+ +#### Resources Limits - Commands + +
+Check if there are any limits on one of the pods in your cluster
+ +`kubectl describe po | grep -i limits` +
+ +
+Run a pod called "yay" with the image "python" and resources request of 64Mi memory and 250m CPU
+ +`kubectl run yay --image=python --dry-run=client -o yaml > pod.yaml` + +`vi pod.yaml` + +``` +spec: + containers: + - image: python + imagePullPolicy: Always + name: yay + resources: + requests: + cpu: 250m + memory: 64Mi +``` + +`kubectl apply -f pod.yaml` +
+ +
+Run a pod called "yay2" with the image "python". Make sure it has resources request of 64Mi memory and 250m CPU and the limits are 128Mi memory and 500m CPU
+ +`kubectl run yay2 --image=python --dry-run=client -o yaml > pod.yaml` + +`vi pod.yaml` + +``` +spec: + containers: + - image: python + imagePullPolicy: Always + name: yay2 + resources: + limits: + cpu: 500m + memory: 128Mi + requests: + cpu: 250m + memory: 64Mi +``` + +`kubectl apply -f pod.yaml` +
+ +### Monitoring + +
+What monitoring solutions are you familiar with in regards to Kubernetes?
+ +There are many types of monitoring solutions for Kubernetes. Some open-source, some are in-memory, some of them cost money, ... here is a short list: + +* metrics-server: in-memory open source monitoring +* datadog: $$$ +* promethues: open source monitoring solution + +
+ +
+Describe how the monitoring solution you are working with monitors Kubernetes and
+ +This very much depends on what you chose to use. Let's address some of the solutions: + +* metrics-server: an open source and free monitoring solution that uses the cAdvisor component of kubelet to retrieve information on the cluster and its resources and stores them in-memory. +Once installed, after some time you can run commands like `kubectl top node` and `kubectl top pod` to view performance metrics on nodes, pods and other resources. + +TODO: add more monitoring solutions + +
+ +### Kustomize + +
+What is Kustomize?
+
+ +
+Explain the need for Kustomize by describing actual use cases
+ +* You have an helm chart of an application used by multiple teams in your organization and there is a requirement to add annotation to the app specifying the name of the of team owning the app + * Without Kustomize you would need to copy the files (chart template in this case) and modify it to include the specific annotations we need + * With Kustomize you don't need to copy the entire repo or files +* You are asked to apply a change/patch to some app without modifying the original files of the app + * With Kustomize you can define kustomization.yml file that defines these customizations so you don't need to touch the original app files +
+ +
+Describe in high-level how Kustomize works
+ +1. You add kustomization.yml file in the folder of the app you would like to customize. + 1. You define the customizations you would like to perform +2. You run `kustomize build APP_PATH` where your kustomization.yml also resides +
+ +### Deployment Strategies + +
+What rollout/deployment strategies are you familiar with?
+ +* Blue/Green Deployments: You deploy a new version of your app, while old version still running, and you start redirecting traffic to the new version of the app +* Canary Deployments: You deploy a new version of your app and start redirecting **portion** of your users/traffic to the new version. So you the migration to the new version is much more gradual +
+ +
+Explain Blue/Green deployments/rollouts in detail
+ +Blue/Green deployment steps: + +1. Traffic coming from users through a load balancer to the application which is currently version 1 + +Users -> Load Balancer -> App Version 1 + +2. A new application version 2 is deployed (while version 1 still running) + +Users -> Load Balancer -> App Version 1 + App Version 2 + +3. If version 2 runs properly, traffic switched to it instead of version 1 + +User -> Load Balancer App version 1 + -> App Version 2 + +4. Whether old version is removed or keep running but without users being redirected to it, is based on team or company decision + +Pros: + * We can rollback/switch quickly to previous version at any point +Cons: + * In case of an issue with new version, ALL users are affected (instead of small portion/percentage) + +
+ +
+Explain Canary deployments/rollouts in detail
+ +Canary deployment steps: + +1. Traffic coming from users through a load balancer to the application which is currently version 1 + +Users -> Load Balancer -> App Version 1 + +2. A new application version 2 is deployed (while version 1 still running) and part of the traffic is redirected to the new version + +Users -> Load Balancer ->(95% of the traffic) App Version 1 + ->(5% of the traffic) App Version 2 + +3. If the new version (2) runs well, more traffic is redirected to it + +Users -> Load Balancer ->(70% of the traffic) App Version 1 + ->(30% of the traffic) App Version 2 + +3. If everything runs well, at some point all traffic is redirected to the new version + +Users -> Load Balancer -> App Version 2 + + +Pros: + * If there is any issue with the new deployed app version, only some portion of the users affected, instead of all of them +Cons: + * Testing of new version is neccesrialy in the production environment (as the user traffic is exists only there) + +
+ +
+What ways are you familiar with to implement deployment strategies (like canary, blue/green) in Kubernetes?
+ +There are multiple ways. One of them is Argo Rollouts. +
+ +### Scenarios + +
+An engineer form your organization told you he is interested only in seeing his team resources in Kubernetes. Instead, in reality, he sees resources of the whole organization, from multiple different teams. What Kubernetes concept can you use in order to deal with it?
+ +Namespaces. See the following [namespaces question and answer](#namespaces-use-cases) for more information. +
+ +
+An engineer in your team runs a Pod but the status he sees is "CrashLoopBackOff". What does it means? How to identify the issue?
+ +The container failed to run (due to different reasons) and Kubernetes tries to run the Pod again after some delay (= BackOff time). + +Some reasons for it to fail: + - Misconfiguration - misspelling, non supported value, etc. + - Resource not available - nodes are down, PV not mounted, etc. + +Some ways to debug: + +1. `kubectl describe pod POD_NAME` + 1. Focus on `State` (which should be Waiting, CrashLoopBackOff) and `Last State` which should tell what happened before (as in why it failed) +2. Run `kubectl logs mypod` + 1. This should provide an accurate output of + 2. For specific container, you can add `-c CONTAINER_NAME` +
+ +
+An engineer form your organization asked whether there is a way to prevent from Pods (with cretain label) to be scheduled on one of the nodes in the cluster. Your reply is:
+ +Yes, using taints, we could run the following command and it will prevent from all resources with label "app=web" to be scheduled on node1: `kubectl taint node node1 app=web:NoSchedule` +
+ +
+You would like to limit the number of resources being used in your cluster. For example no more than 4 replicasets, 2 services, etc. How would you achieve that?
+ +Using ResourceQuats +
+ + + + +# Hit the Star! ⭐ +***If you are planning to use this repo for learning, please hit the star. Thanks!*** + +#### Author by [Harshhaa Reddy](https://github.com/NotHarshhaa) diff --git a/topics/kubernetes/exercises/kustomize_common_labels/exercise.md b/topics/kubernetes/exercises/kustomize_common_labels/exercise.md new file mode 100644 index 0000000..a215a83 --- /dev/null +++ b/topics/kubernetes/exercises/kustomize_common_labels/exercise.md @@ -0,0 +1,17 @@ +# Kustomize - Common Labels + +## Requirements + +1. Running Kubernetes cluster +2. Kubctl version 1.14 or above + +## Objectives + +In the current directory there is an app composed of a Deployment and Service. + +1. Write a kustomization.yml file that will add to both the Service and Deployment the label "team-name: aces" +2. Execute a kustomize command that will generate the customized k8s files with the label appended + +## Solution + +Click [here](solution.md) to view the solution diff --git a/topics/kubernetes/exercises/kustomize_common_labels/solution.md b/topics/kubernetes/exercises/kustomize_common_labels/solution.md new file mode 100644 index 0000000..cf66800 --- /dev/null +++ b/topics/kubernetes/exercises/kustomize_common_labels/solution.md @@ -0,0 +1,31 @@ +# Kustomize - Common Labels + +## Requirements + +1. Running Kubernetes cluster +2. Kubctl version 1.14 or above + +## Objectives + +In the current directory there is an app composed of a Deployment and Service. + +1. Write a kustomization.yml file that will add to both the Service and Deployment the label "team-name: aces" +2. Execute a kustomize command that will generate the customized k8s files with the label appended + +## Solution + +1. Add the following to kustomization.yml in someApp directory: + +``` +apiVersion: kustomize.config.k8s.io/v1beta1 +kind: Kustomization + +commonLabels: + team-name: aces + +resources: + - service.yml + - deployment.yml +``` + +2. Run `kubectl apply -k someApp` diff --git a/topics/kubernetes/exercises/kustomize_common_labels/someApp/deployment.yml b/topics/kubernetes/exercises/kustomize_common_labels/someApp/deployment.yml new file mode 100644 index 0000000..685c17a --- /dev/null +++ b/topics/kubernetes/exercises/kustomize_common_labels/someApp/deployment.yml @@ -0,0 +1,21 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: nginx-deployment + labels: + app: nginx +spec: + replicas: 3 + selector: + matchLabels: + app: nginx + template: + metadata: + labels: + app: nginx + spec: + containers: + - name: nginx + image: nginx:1.14.2 + ports: + - containerPort: 80 diff --git a/topics/kubernetes/exercises/kustomize_common_labels/someApp/service.yml b/topics/kubernetes/exercises/kustomize_common_labels/someApp/service.yml new file mode 100644 index 0000000..4b6db34 --- /dev/null +++ b/topics/kubernetes/exercises/kustomize_common_labels/someApp/service.yml @@ -0,0 +1,11 @@ +apiVersion: v1 +kind: Service +metadata: + name: my-service +spec: + selector: + app: nginx + ports: + - protocol: TCP + port: 80 + targetPort: 9376 diff --git a/topics/kubernetes/exercises/labels_and_selectors/exercise.md b/topics/kubernetes/exercises/labels_and_selectors/exercise.md new file mode 100644 index 0000000..d8600c2 --- /dev/null +++ b/topics/kubernetes/exercises/labels_and_selectors/exercise.md @@ -0,0 +1,11 @@ +# Labels and Selectors 101 + +## Objectives + +1. How to list all the Pods with the label "app=web"? +2. How to list all objects labeled as "env=staging"? +3. How to list all deployments from "env=prod" and "type=web"? + +## Solution + +Click [here](solution.md) to view the solution. \ No newline at end of file diff --git a/topics/kubernetes/exercises/labels_and_selectors/solution.md b/topics/kubernetes/exercises/labels_and_selectors/solution.md new file mode 100644 index 0000000..def344e --- /dev/null +++ b/topics/kubernetes/exercises/labels_and_selectors/solution.md @@ -0,0 +1,13 @@ +# Labels and Selectors 101 + +## Objectives + +1. How to list all the Pods with the label "app=web"? +2. How to list all objects labeled as "env=staging"? +3. How to list all deployments from "env=prod" and "type=web"? + +## Solution + +`k get po -l app=web` +`k get all -l env=staging` +`k get deploy -l env=prod,type=web` \ No newline at end of file diff --git a/topics/kubernetes/exercises/node_selectors/exercise.md b/topics/kubernetes/exercises/node_selectors/exercise.md new file mode 100644 index 0000000..1d588b5 --- /dev/null +++ b/topics/kubernetes/exercises/node_selectors/exercise.md @@ -0,0 +1,12 @@ +# Node Selectors + +## Objectives + +1. Apply the label "hw=max" on one of the nodes in your cluster +2. Create and run a Pod called `some-pod` with the image `redis` and configure it to use the selector `hw=max` +3. Explain why node selectors might be limited + + +## Solution + +Click [here](solution.md) to view the solution \ No newline at end of file diff --git a/topics/kubernetes/exercises/node_selectors/solution.md b/topics/kubernetes/exercises/node_selectors/solution.md new file mode 100644 index 0000000..5b303d4 --- /dev/null +++ b/topics/kubernetes/exercises/node_selectors/solution.md @@ -0,0 +1,29 @@ +# Node Selectors + +## Objectives + +1. Apply the label "hw=max" on one of the nodes in your cluster +2. Create and run a Pod called `some-pod` with the image `redis` and configure it to use the selector `hw=max` +3. Explain why node selectors might be limited + + +## Solution + +Click [here](solution.md) to view the solution + +1. `kubectl label nodes some-node hw=max` +2. + +``` +kubectl run some-pod --image=redis --dry-run=client -o yaml > pod.yaml + +vi pod.yaml + +spec: + nodeSelector: + hw: max + +kubectl apply -f pod.yaml +``` + +3. Assume you would like to run your Pod on all the nodes with with either `hw` set to max or to min, instead of just max. This is not possible with nodeSelectors which are quite simplified and this is where you might want to consider `node affinity`. \ No newline at end of file diff --git a/topics/kubernetes/exercises/taints_101/exercise.md b/topics/kubernetes/exercises/taints_101/exercise.md new file mode 100644 index 0000000..a5c2e52 --- /dev/null +++ b/topics/kubernetes/exercises/taints_101/exercise.md @@ -0,0 +1,13 @@ +# Taints 101 + +## Objectives + +1. Check if one of the nodes in the cluster has taints (doesn't matter which node) +2. Create a taint on one of the nodes in your cluster with key of "app" and value of "web" and effect of "NoSchedule" + 1. Explain what it does exactly + 2. Verify it was applied +3. Run a Pod that will be able to run on the node on which you applied the taint + +## Solution + +Click [here](solution.md) to view the solution. \ No newline at end of file diff --git a/topics/kubernetes/exercises/taints_101/solution.md b/topics/kubernetes/exercises/taints_101/solution.md new file mode 100644 index 0000000..26d9ca0 --- /dev/null +++ b/topics/kubernetes/exercises/taints_101/solution.md @@ -0,0 +1,30 @@ +# Taints 101 + +## Objectives + +1. Check if one of the nodes in the cluster has taints (doesn't matter which node) +2. Create a taint on one of the nodes in your cluster with key of "app" and value of "web" and effect of "NoSchedule" + 1. Explain what it does exactly + 2. Verify it was applied + +## Solution + +1. `kubectl describe no minikube | grep -i taints` +2. `kubectl taint node minikube app=web:NoSchedule` + 1. Any resource with "app=web" key value will not be scheduled on node `minikube` + 2. `kubectl describe no minikube | grep -i taints` +3. + +``` +kubectl run some-pod --image=redis +kubectl edit po some-pod +``` + +``` + - effect: NoSchedule + key: app + operator: Equal + value: web +``` + +Save and exit. The Pod should be running. \ No newline at end of file diff --git a/topics/kubernetes/images/cluster_architecture_exercise.png b/topics/kubernetes/images/cluster_architecture_exercise.png new file mode 100644 index 0000000000000000000000000000000000000000..d50cba76979e085aa2dc09ef43c75b0f2123493d GIT binary patch literal 46202 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 literal 0 HcmV?d00001 diff --git a/topics/kubernetes/images/cluster_architecture_solution.png b/topics/kubernetes/images/cluster_architecture_solution.png new file mode 100644 index 0000000000000000000000000000000000000000..535528d6951a78105b1c9b622fafa2583f5a633c GIT binary patch literal 44343 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

k5y-8sUAc6uuWdk5$(3nKJ?kP-ZhG% zx!?fY#`p^l8}j5-i9IxM2}&akIv;E4H|vb0^|w!>d!T8?bm~hl3XTjt`*mDEX~dD7 zS?S7~a6ItE4ur|=Ufa}SAw?c39yyW6W;YL*jD~K&Dti|7VApt1)jf&DC}|?4ppcM0 z$bM;&YA;8afTs4)@0DHpm7fin_u^i~09d$fwN@)w8bl%C)osPvON#K~8J=zRq9jZA zxe`-R*lv#%xj;(GAfP0gbV4W2P>;X~ul6-k-G#JQ_lS7mLY7jnara}~;c88-wWk7{ zDwXoG1Okk@p-=}o72r+Couc9p<~|7(<`_E%LScfGf$k2|G;b#_tL-Mpy%}R{p%*** zgZp_`>{MK)b2rH!NaUVt)&%w_!W6VM2kIi7tg)U%n$)uo($^#VS;cxL9Nd9#oBXJ5uSRO>}L_}q;Pi}toF!R}i#@aY#G z{Fzm7j|H^Uj^lWFTS=M8(_8w_J{(lpt#L(dX1%!PC+ZK*vLD4x%CFVpB!{oF*V?7U zDmYWEo-2UdN&7%=+s1_lDQnXel4bEsxF%Ft7<4dis?im1y03 z;BQs07+2y>eukcO4;c$!FMPWVTy|E={z2+3K)yuL^&fnvW0k&ReO`Mw`c?MJcx?}y zv7;{_mlOFrb}UuAeX{D96VfEelQCL|R_&Nv5t3hOWj-IM0A2CuG#btZY>0<&J*FJe z7+-Bx8E5VLpI(3;lQ-am^*%R=o$i#cqD-S+0)m2!qe10rI+m6^wCu9+nD|S)$FU)` zL!4zIw%ufBL)XVe)<4wn7iF;6g58WCO+Sa(O|U+l7}kHpI#o@hG}b5-{K|ztflVe3 z)&B-;n$Jy1(&lrKBykCMwvZpS!$RgGztkdQLMXGPsBEEEAr@MM00}~J+!}bAM2ZRA(TY)mW{tQhbzUq4abqyxlY;JbF$hrakm&`d_ zGHY$G+^vlrPkwE??i6I7ovIl&*8v50kpI$PDHJ_HS+zysEhvrDB+*eUOJ@z__RchU zu6^>ID+Pp{e?utY-8G6zzn<)v=#zct(qMN$j|6PibiW8nELI`k-D#)yv{0@R=&qr~ zj*@E){r<6W?`Cw1{jdOF2FF6>qPXAP>`CloY0@cBra(Sy`vj4Ab3~jo^U9`XGVQpuqNotb8lGTwTcPgvMBExEjM5a>)z2;t zdJj)J3`o-i%5jjt`L)b0;mK=#o1*gd_SburuQ0BY$|`0tid!#=A*MSzci@~Nk`iBx z3QOFE3>GFm?IxFAm@~?(A;wvHX#2jXES5l|Kep@O0M)@zyU4y#z9BXna_C|Xc(bn` zYEbV(N<4xYB{!1Ce2qt6Hbb%+=*QIz5k!XMaKnpc@Grtrzc>U1>~%4C1p0}dzPNny z!)C+catR}m>xUdv&OCxt6eCY?n!*e0r%)lmdpwwhUTKfPDH1?y?`;i*VJgx(I(RUi zzbYUu%iUuM+NgT)8;9;^AyY z%e?=&$6s8TYi*hYYnH=m>*}ULsits6J}fodWOz>EKB}(~>)fTRGJZd`i>y;&iRCKW12c=9Zb-(C-gt>bKJ9<*De^b#H?7hbzIrwUz~hy2=Kw3~t)dM*enBBC+;M z_Z4!nxG>h3+{*AD3}F2E>~HU&Lz0)ACM%0xt?{~Eye4sXspF}l2BC}ycanxdg=5IW z%|xx7qCc9g7EKYdnq#!ga9*)N*MZH0utMr2@BiIa+niG8gyzDPVGG|+s!;s9Lh$c= zAoh}wbP&h}XBHXxGASHy@xm<$5xP1z%Vff>yu4s5tty9n`-s+f)S&X}!4PiX0~PS| z*H5EGU3(}fi843urQ^>p#4ia~P!@0}b24y@pO0^AGJ^@(!!*e!x9(<&*v9lD zIbPywf}MyiV)mnM%LO^PSw+0+Y$gRViak6hV>;LG|F&Ltr2F0^11!FHSD}!DoR2{_ z1d;OQz(-;&FabKm5T%3{P9u+;;yLR1^+f`HHEHqZj zA1-}r_ALXsUS37f­|9I&Rl6@_@T_672tfx!NI+!9}d8slfHZ*Z~HFyUQoJpeT zsMCr7Dmtn3*~%C!d|T6C(T_ha{H-nJ+v2QexiVahzt0J}PAoIzeYb{)XI&f`S_IXY zkqag_iwG^m6mQ1UZBFxuaKBzny7nFN;Hj=9liDBVwmF2Q<@vV_?hA@Oo0pt4lP5Xo zh?2qK4m=U0Xo$d>Un07Lq;6awJ=+5j+#~9Ci_GVK>aS<(u{;R#Ub@(`o9m!hWG_wr zmNI4u%N?BFpvU8KJLH$0)v{eYTtwvVbLXjt(74eSz{~zl-tEdNr3KaWJ!O?FIB{J? z`S&Wn0@HXLh}{l3rhqPtO6=~VWi|89^^+8ox%U5FN}92T|bsT;lv9$q|~S8 z0gfy&N6u}~o4`|^7v+P6>cfjkYFEEwk+o zHPdO^$~6#q>6$JOUxx-PTC4Efr?a&Jl#!uanT@U2ae#Heduj zt{y60v;Hfb&!M9J?keS=7-TsOF_5ejS?yov|xn&yKvQFLsuSS^!h_fYvk-_>5z<#3LdgtK6H1cWN;O#ou z93h%!L&}F@4M*EnT50h#Pcb#Yjx5bA$8e7do!z-H=&rDZkbK` z;U2cNCpgT$PwD4Od)iyi^1cgldV4bwsf^VWpXVUtzCsXwtdx#mN+FRpHW`{)9F%y- zuwSjpkka6`;uruLmJZCSY{MprPjA{NozfgFLb1^i1!p<@t-gmZZl5-hS*2 zFG2pG^5R22Q&G^SDzG{ArHgQQM15xf^rp>9OEfL}O990&#t!N~;Otr085EkX z`GIcifgBlTiyU2eg<6BL;jV*lRQd~ZO8ddy7Z=e~f}AQht*ZzXp&}p+S#s0Ao&VW;x!?s`ZIG)K5TU7`$EG|iYq;=vw1KCCLpf*J ze4ULf*p(SWu%$KTE4f20qb_Mg74ylsI!s+%D0Wp;UcSAP>7CzMpW9S!%- zLND_d42BnHs%v5(P`2|7FV80nCtoQQUIo6VC&z;^@U*~`zky%V(|I79F!}WM5H+J} zWn`FMOs+RxjNcjk#s(70X%T zkpX;@Dy`+HLAJX4dg*twNgU@YPf(>_IZM8RK9tldmRpq%C;;VPTIDeO+#>5(q>#L*8{q3 zk`T4Qpq-dw8kd3XeYR=JVCdu`CX<2FjD4=hTo1aB_a<++HIX;mbmBc#n1j?c);`c+ zN`03`ui%M(sd_%F=%Z54r38no2ivC^>dPPev`(m2_L$t~`Ab(s`R99;0Wo?#E}4C4 zTpvgKeRbY9nNu}6nfX|OM|F$k#H*vaBuMcE-E#SsL6V-SgJ7Hk=ex#-Z~bv!iK;<9 zGrdoXw{BZ7Gbc`#e@OhU%(s8anf{rK$L0bC;FtS6x93t1<6u!FgReXp=&ziTu0z^R zg7ysfjr!Ooh##>hThV&Wf}Wy&lYFUL&&hSXVV^nGV7ln3#PQv0mMCn8gT%zV^I++h z89m~LiYL<~EY_OuiGlih=UWCZU*d-R#tksLf5%d%R+7cz&<9ALz3vYcN<42#Z_CRc z<;m-{WN{IHgzfS0TK!m`cs1QvmcUy;f91uSy-|&PS>B)iq@?08xffri2#Fw530+a(aiV+v31XQ955A0nBCUerL6*kHQV(fl2>?{(do{@bvSDjCuaf^Q5 zq98bM^}llE$`ryH&5S)oA$juy-BFlqhisv@lQB`qZd?W}8^CyYN{B5uMSnk!GfKpfFw+~IoqY|u z5pkgJ?1;68;Yme%WhtajiKksU_fAU9lvsH20*YR#<5&^o>C#S@LOwlB<~+H1oSvS} zm&1NA$HU*8Gw;XF>QGo@fjQ&bX;0NlqF2)*MiygNx>Hu+^Qcl`;4W1^pF0r~D*Wu+ z7{|MB#v@socDIiGz(Dt^75d;opy{EKHoUIDyt;V{urDLnCJyxU8i)C>0m@_f+Tbq{ zjRhqFIdD%r<7i~N;YW*zweGSz%|`PoV) zaNoIF?aP4*vXpqRSor+H=9qhnXXX68`ZY)zsqKVQOFt3s45Bi4(V&yf;#&Vs>e%X&Wy&Vk+V8No z$V|DGVR+m^=Asy!N4bKFd38@&EAPCA^xON35@o^7zBR_!An$rU`%fF-$gYyD8xT6aLV<{c!_Ha?8hD!V_m1I?MtJJ&eJl6@Vh#aU$`<*)- z;fWC|Y@juhDu{)J&2aN|>hu*%Nd!%;oJs%n!J%>h*zP5(c-*O-d3AwxdQ z4whUY2Udy6}_4C+k{8? z;D)**i_^0r-naI{B%isN3dhBS6dl{<+_}|NU!1qqN_5EYX5(CvK*3mNNYI-JKrc*_ z^tVIMCaS8c9;%jD#A8aoKd3+F3@)VF!VSK>gnDH)JZ!Jfvu(bT4*END@emv-*r~s8w8oPYQle@G%S`DD(?3xL5 zpOgqi&bEoV2+yAXt&}}pYo)qAtB@tT z*}L?HH?WRnATd>nHKh_4my??-)_J|ZWwCPPwCCXcX_F-fb$BcU+csxoW0URdcW+64 zKfJLZi)%tdPzf}a7lLs40*5hFvV6Iis90`Lrp=)z^zb?vo`d*%s4riMXH|t)7IM;W@UC5BG4-im9NhKV=pYgDULqi7tW>D@~Dl;uNB%sJM#_&g2&3&Lw3caKWND?kHL{dfsg(K?UZK`LsoBzD#)WaR3R zFN>I|O5*}FzJK?l$FOJ#h8GS%hK~^D3N?SnyI>LnmUp`homOz- z@@=p2yG+v6dK~EpytMSLnMN8YNsrcj7cMb|KjGr&VdIn+H zP(55cXJcp*Q;@ij++y2%estl)wj)4|dlms!?(052PNS~avOB04B#xuwnCpsXMg2c^ z=Y>&bEd;qj(`KJfCu)GvDawtk6b&7ta}*Dg?c zcqlpdGNGGh9(X7GeQ&8i?iP?S?d^YbyXavW9gTImlhOF|{CTP;oT=PFJBUFop@kc# z2h3gR+cT7`73AT0@XC|J(=ZDt)Oc8i4tE7*#GVzfv^I!u27bzivob1N5$l zE~83(-yBcAo{GtRx{z>9Ix(?hr_msdor2U<6_9;9X6WO*NGkf5U!PS{MN3*lS){ZN zT@$cZT~P7BO_CElKLY-U;W+lCzVnXoB;0VZgXI-{Lhba%H3Y7+(;@x;p*4F9h0U#R)lPCM4iw*Yy)kMp9iam zRTzp&_Z|YrlTwMp)7OQP*#(X%n)I;hMPm0XI<6efgS;^+L<*uGw*62v#?H;U(g)X7 z=L&joqn9~$mv5GczX$8_NJufMHht+uD4Ha590jaX&UejnMXKJm2eGL-Zzo#k(_{X2 z5%CXi!rpyqceD^tFyx^5L}*WQ`Ye!O9avsqXZsMP)8*J5SP5E+SewCX3e02EJbYL! z=j4s9LF>}sLLbwvd)gJUoaLbG==Ql~1GpYyrz?RkyRVs|i-PdiCiuGD%I->bLz90#u#5)|vUXmu=+* zE+>fgT#J7r`lH5OY1Wiy#53ulz|<3?{JUp~`VN9h>W6*|Dh*Ecob1Y42W7qjSrw%* zW3ad3;O({bb^RMbt_`TcbHnqIkLXB;-n|pUxavl6Lk_d%l{Z6rBWy?>$-e&+B zt1L1QUz6kQp+QWlLEL~OI8!MlUGxwMx1D+60kb_Yr3qB78qa5>p^hL6m+h8fH8%8< zQE=bWubGx~H+LO*#wX}HAUvuJp@+#bB=aN}jS76}WH6V!ny>pF4@y%(tL%cqo2{W# z-fuzImaAdo3_YLf{?;cA98(yI@is%t572WzJzJ0OL}?Fo4gh4B@o^b+#fu?bK{IjD z3WM0x^9w&oIWoTG#jxsmn0EDN+Y~L{mRkK!vr+Ap-#z?E%+X8iG|`y~bql;JrH?(J;N2V5BbR6Si9(M7 zgBwr>Ua+#82oB?G^AkZI$ibT3%`pLVZhe3d`bFsq^80#^MT`b^z~6Ir)EMES!i?za zX0IBtp3EFv4Y{nD)k^2i;2?f;;*Q?iHy%n5wV8FOT}V6VtYbHQ08K!u6sq5wjNtb& z>?V2nIhuWSHgLNXGyqPmer>~7w#U$q#c+p|-oB1AL zwgbgYGrr`U!px;JcP3J1(o2JHzk1(yXbJG&QAFvA|D9j_kBiX3zt`z}m9K)b_EWjn z^|-T1*A&$|Z<~C+@0gZ`^!q(<)c-h%zUOQ$H!3-5D*O4FoQEYInHSIJML0pTyzUD~ z7PX~Mmg#UyaL{;PmAy9-A*Yo7;S|`lOv}u270Dp?@psNEs;lU7(l{$cd_Va~`6X`# zm7Zy_$%cMhWUa(uI?&o>p;8&t)0A^|vPY^B@G>_I1Sn==nW?M02r)xdujzFQFAZtD zqR3eQg58*Csdv)*OFu}K-;I1`1gILxf%+r2H8u*v-!7&VL7#hae{IgA4!(Av4X+AF z@hAwx!zZjDN{!~2 zR8=sHEMh~qX`@Tm(&qkC-4KZ{n?Z@+kAFuCH~NjdJ$;=KH1OJQbZ@-y{@hE z(e3;v4g=~5cnlq0fDT>(D5nU{wlfr{)q62Lxx3IJPWj6ZYrhq@_7wB@aZyhio`jQa zgaZreW7Ad2ofRKk%G+cQNJ0LAI+GTf6`MnMn$2xgy?&bR`&`j8>ke zm7fg^yspWb%9zw2YmXeN8*%pPhN;{)z9G*ol}dKUr;Gowhn8HM=Oo$YV*Io({V%ac zk^{sRv9ZQf7GJV71%Ewii&TSY1<89Tlaw0M^Rjt%jb$jJw29p&8Y44OQ-eK0pc$i4 z42fU%n!k4xyA9cxqKN`iNi(8*1+S~xe7QxUeXTHR(m4*13!$mO#8&pYliB~$5>QUu zAdK{4JO&0=twEEmJc)IQ=Bis__-D@VA0-fCa;f?=@oRz6CFXI^VF@_mud+Q6ASHvVZUGcAmDyP`mihdwBDz~__)Wk^_5%mxtvmO zsaK|)+Dp5#z8~v>1=daph-6K?HuVk67t~fvtnx`;lJj#*vA4>IHpg0$> zn=$)F75e*?tn3Dpb~!E;uY$ON{UP$ZKF}XnfE>jtfSdIkTK|f<-f-vy>$Xf~TLGA%Djj`fup+3i5oMQj1592RFxK3Q7xq>8(?g=<_^+H$!j2>dXD zMQzNrhHYS`cpe)H%I=%A?NR(7?;Jq7gYL2i1Em0H&Fz?*$KdF8(IlgKt5_v$PTGQ( zz|#Yq#osi!q)oyT=^r)5qR=Wodgv5MD5+At%@}$c;}!jD z6xcav=;~__!tDRD05J!C$2-vMy0;jo_-jyzG7l)anf`WJHPkQMBqvXG(DM2OU7}ZF zU>axt%M`(F(5gH%n^x8QYkIorXx`;$T-LQb3M_Gcsyn7s}fWuXsX7Bmz7Bqb!Ez!N) z_|EK{yReIsdhJMr!OXqrSIjv~gla*`r?tS)15=Y#X>P|^h1^6)W$K1IH>BYyS%_l2W;;CJdZ8wx^xVpduyVRGT2all?6!9VSl zC17c+L{CWiv`FBk< zNq2_tT$3)bVNxEKb;cXuE?#mXzL5q_glp3hTvXQ8cQ^T+_Y3-RCU&z{|vS6|z2&EmiSK?p) zWKbEirg-49v~)v6XcTpbFlx4>v(oqu6CJq31S=?*nHXP!Fr|X$#g^|^UhTY_WL|x0 zZbT`3?88(QA+$7hbEG|d_eb+6f6RADeR_bxb2_WoJ6BIDqtN7&lrCR_BojP@r$g!< z6Q&}bEbh6KMlWrZ7C(FRA?3xlbRJue_0IcK#qkJ?=Yoww48&;7Fi6?2DY?Ufw7CTV z(m={%|Ev928jjup9(8S7YcR&%mjK+ljh%yaGa&FT;C85+HxW}|vpcNT;;_4!%4I74 zRy3>v-mgemGOH9T2m%q#Z!{nqw|=gVx#GL(bA^qeCS)$gR7qWGgqC~;@WE1$GArBn zV-M%XWzin>C>nz+?gkmce^qCUn(sTzmC*uw1x{V~&L+RsFM>Y_&e|Hvtok<0*m7({ zYb$@?6_TzHIFX~pxdMp=e<8V`D+X_d$Ir+t{Bz$?;oR?S@xw^1$i_6T@LsA*_g zbzg>|%b{QL!$0x@#$@ALKFXrM9;zi9oiX7@X7O9~2${i{>gDnC@4<(^Ti|P7(gYwI z3nWE}T^d_!#^FvT^KognqC9t>D{hF`)n8U;%IV`p)Msi(&|{5$g2NQ*4i!4Q*!%4H z!~DEi3dX;vw_XBdu+fuRPZbPIquHHXY(zzI zNeRJ-9h@jrlROF?Gig7_@>D#q=NyFkC11kGAt@?;Ur0=sRY^Ek6~>CvMfvq>I+u08ze@c?M#G) zTnew>RG$iDdIha1bSn~)ck8XRcw~K;q~l~?nA#%REu`*gFvD?aU+MHz#;Xya&F1@y z`409pIIV0q+F!BF?dy)cmhFK@zCG4~@XJ%-sx@KH{^hUIdQaxay0`Id3k_PLi5bC4 zL&lCSGJ3vk^fAQxs|Fo=C|lLOVVc7+fCp&fcwQ%@r|So;9RB>;9Jj!b-0i5=g2PPa zRQ;Xlzti@YkjVWh9&#cbi9f~KydnWwZ#33+08wlqHv-`e-wAQI;|_m(BzRw_OQlSM z4u}w}obf--SWaY*(Z5Ms`P=pMNg_e9hyvF#WR-YSXOg}HEF<7w+*3)0)J|Y7Nla^i ztReUo!q`SMCqwy)vHz;3Y+Q`YeHx@GWTD}Gcpm?#_|R7k7-xJs%$EFkT~nWw#SikjJ=aqAaB> z??x=*Bkk9=ficKwuoK#TxZzn{Y8@F%N?WU}?C)aT9K$o$t7_W&QI;sHXj&vWUVuEP zY$V>Ta3@AahhT>edZ|VJt)wyE20PLJ@KG0=)Wa^qO!52Na{HeL96vY7yXUdROsF~SFW9L%Cl4o(9c@8_JP)-E)ojJD zelSkr-yl!R@*_h+gI~(}HkFEJ<1QAWfnnijEsBKE=vd0CqtS}N^OEVE@ELoN=H-58 zFhzmDSM^}=?KtNP*0b0N3ySLjff1HS=tWv1gVCJp_J?bI-ET^?K8ED%M#op0oGDOU ztt}$^Qk`_88s3ugI-db!s5a33_hiqkUGXQ3P?_sLV#A?bD*hKsZKSbF@WD>R3OnX2 zrPh|X5`{r`eV9x%Tsg{bs_%V%XzwTzD_y2eCFPumKI%@Lwvx6GumR)=mGV~XIiS%@ zuq{vko@PE^H#q#M4}6Wk$0OQ+bhAgJly!2))dnk{0Iev{0Ff@5SolAuDGz1V$h&ev zL1H7vIpTv9{RojT|8e7%YZ}Opn*NgQXWhO_yfsT46_NGkJwtF;dl0B9SM`9LF(ODOxh$LNX+l02 z!P`kM;PD^EMP=fL1=Z$D)_du5G*CqZ(1A2xyIE_o-rt;GX*FRaji0}6wuw!c*T=CSx8Uv9svf8>M{MgI@;B;74o)NBLT5DXW;V?3cWZV)r;t_D`f?oBC>|{)3i_b>cd9f#_M#0 zH~v^}I;1i9JFvgV-hqsKoeHPIU+BH?)uU~eIAWIZ(@TaAq2F}$$x?pqmj*?GtTlVb zInHifso~czyy?|Yy|^jX!v*{lfMry zI$1y!&x=+~tcjVYFaF%)p|{Xafv-~_)8M6c8BKH2_hg&XmS6G5)+h7oOG7taINM~5 zU{P|q^y+-CoOW#U>Rf_$x(W+BbxnR=dcj)C#(&oKW1X%fj2lehn09F z+$w3(ieIi4q<3UJ4GkTOTQq`C@mQBQ+Y29owagg6GWC5IWF?1iJ}1d@RpH>5BP$gr zvV~iZ$v9ag`w~xOJySa=vM~(*#6ctc@1_-Te&mUt2vsOl$yDu~dW*Vyd+0~_mp|Bx z#pNP6rFf{>5@B3Wo=uSFA71pKTKF8zJx6lC_HDHspT z5zZ#~P>VFnI9?ghhPdpmyVU+HaP#6l@lpgxzc^2erA?iSn2!ip4%~fp*^?NE!NdK@ zvU-V(@_u@tfQ_&Hqsfqng2FGdeggz6ODXHJzv4s~bG;UAY=`>har^|or^^MyPS0*8 zO0(Ps$lN`+F{^los7mk{S36ZM{q1XNM#uTVADpF;%fWBe>!6DC(P!DN89OsYQbGdc zqaW_Bo}(mw01mWQA65ntjt^-8Pw(=lX&wmg+}g&9yO&snvnAw%b&?THzi5jxFb1yfze z&RH`LT^Cv&saAxqL;i4(mh`#+Rg;1bS|W#f`F>91s?vXI4^?RNM;!QUxt>OAk~&O1 z)@4Dea9iIz2GD=K`Z)l)t=qvA1*W;faSqg)WXW-wg^x${ix83KLQX1l=!WxI!c=Od zZe755HmltGl2Blt3B0}fTO&IY(ny4dCk;wg%!d^Q8FgAm>E1#|%gLaXAJU}7$L=Ur z8hg*c-bND zRkg?Pp9swm_-Vn`p~AWE9X(4;XZ4r1CQmTwmv%2}a$0qC(#;m8j{*Rl9MrL-+8&lfxz* z5KjC!{PRr3)R3ids?Ci3*8GHJBK>ZcP3i|f4kebjEbNbZ3q`}R0*myZmG^|Oonmfa z8ISyVpahudWMfG4pb9Y%-5KkQ$xin~fAef_5EtCmX|P{i_R--%+)|=XHY&G$WhG#z zJV>p786PXT;&=2;))#gB`b2jRn(aa*ruWS{a*DS*6{@a8%%iz6@C-aW$spB+;^Qk8 zlFt1uU-4=56y`K^XEXkWo*Ib(T;TW)6I=LV)f^S) z-Er1T$u&G|tc}I;oIgcb3H6;2TiZ^TN zjXuk-$~F611kA7NeL*Zi`rcP9V{Gba-~+bTQDvZkqPc8+Vs^F)6hFzU9J@JtvUXO~ zQ?hnf`-|Cw271p476kAfs}QVupZmnWeR4{px6_|zYvLDfHI1jHrZTzY;!o|qtF7?_ z3I#l)g;-Dvyh6`stWcTZ^09R!2Tk3iA1o-S=uSC@lqKW4f#LP2qzR0*q#aj1DEmuw zO|oX^=k!*0kH_nM!5EVfG%m?Z+7Z~fKtB;;}_GS@ykf?Xqtph<=d4FAQ>lGLFWEE_-{39^MLlyC{SRA-$xbSD$O z{auimf%53g+{MUHO0w9=X={?0kYEh}aA~>!Kd?nyVr9%g@9Y*I8pWAzAix$wxB*oh zg3gC+NK%P{JIjl5t(ht#F&j=!)Oz9?&^@=5o-Zo2XH)aiAqQ$Ni7QfO{8#l>eEfyh17h8|B zl>ER^Q+)NyyC#iGxO4S*KwO?oN-|$5aud)`6QyjkC^pM>3*9^rC`QxkZ*l}`x`i*Y zJtX8}??%iH1VMD)=+Xei~q)rJ(-&;ipw-?&^Y$Eo;>f4+791(sEcuOQ)X*#ax2hDs`tcOjv2$Ur*Kj@qUguI1uaQ-%SUL3 zXEd$+Tn`2I*DuJgXCyo2EiHWdogY2xcGSobX(3ZV;2h?1wsVS*hufl}OMA8r=9ja;*LK{rmJ%lkLxRfA&POXE4i;m3mJhPR=3QF;7}@ z?xcZCYgYi(p@00GxizQ zq&Gzi7nc;e$mN0=ux8D@Z5IEB(P)CCDNiElfDEZTCbuxq<*+*uIcbO|23Zw^{aYGs z*m{Fp`<%`_Fnk=+_)o*s>d{29X>|g*+!j0gC>rQxz2M6X2?^1Vkbn)v>wCvePanc{ zD1i#k=|}6*S-RMxk?)2-Y)2`i9q&{ntI@i1Mbtd)x#;rlPlwcJ1?%_EIucu=+T35g z9x?q+l^P$fK6-xvxB>ziF@{vCw`j1Ihj168SZIO2nam|ZQx?^aT6f7oJxfSdy+;f0 z$-;}s=?z{ZMuQHZC84U6X(*x-sB~ciES*HGM(t_xv3g6d&D{KaFR4NH>);nieh_n6 zq6yAHb*rGvu?ON71waWj1^fpi>10Ae$8M(^k?a~Eyo|KIz;}pk85M@m-gJJrm}s|7 zDI&+WTVW0v38(e$@V;Ujt9tjRICMGqxbHapba&VLCNwHb?D^AQD%JGU zePK%qr(yFrivB4W;?EArOU4Qe?ah&EUU*6*yp}s60LY+4C*<&Z1X;tkNE?4Abprq8 zTikjXY5$Qx-9%3yg;+eoKF=#&7Tp}j8p>uq&^rC-K^>AN#X$1u{96TuD8$c8$*cXC~gn?oCCl=gHCVu>vzo!yoNcl(8exSaS!1Mkj zrUeLHl2+00nnyKHw`7ffmU2YWMStShM_7_ORd-r0RP5gxUkfuZlyaapO8D*1RUMQw z-mDyieUht;m1@#3HQCmEy z-E8sy@FKHy*Kf=0gAwUzlzI5v+*rYfVw-`@)UH(wA3K4MtPRjNzrn4*o=N9adVha5 zA)@NWv7TYy`FU+QNb5Atk2p_@>aED98H?F7jGB^6Lrt*OYIzWdexrGsI)cjm=QjSjy23WE2RBf6ZL&nCTv`wm^8_^vvBPRPA!NRo;6UAlb z{A=N-73|?(9GMI+BA3}-ml4}7*T)Q);JrMEJKMPjP{`{-G(-?i>=&GaX+Y^Y-&yIH zP0_o1Hj%$MfJ#jQC8Gj9JN=i9x!x4RARW6Tqz*oh0!k-YyLCG3Z(aj2WaQ0^y!xnM zr!}V^`?jD*@*ZTF3!$ii^6M~z?q%%W$`X~V;Uy9z3|9k4GDm$_wO&IAfVammlYs06 zqao-9l`3=c{JNY7m0k!dtD?Fs0?LI+ggpXXtu{6x=(ZJ<4Q$gYEV11XJkqwE*Pwh9 z-w1Va-sMpI!v>hH4spNB@j8sOf=fuJJv9PSS{>#wb#Wz3gey=PhHmI&<$+RIY^pTb zyI`4rIUB0eb0s3(j4>Ne{UEXQI_ z5R8a~tpmd9jHv1e3fyy`c_{(jZf2~QBB99d!QxU0@uM=EL5mo`4!7lautcOyLPO1( z3qGsX+{F59c;0q}7M$Ds+H@P^+Y%ywPxJwP-Id#S-Fr*UuU7S5eRKgbI2=uh4D5pc zKd>ucHGa$7e7jBuLYxh_6bpLc12dAzHdT=|QlT)*;2pL94)}?97%gl;3lgmY)Tte( z!_560{!d^-umZ$fsV@B$im_kc^;DJxV{Es%?O1Mt(AozK_2v=NURPQ%4^szj->Yc-0 z+>Q}znBj28%&3q1jdk-$>M$1UvF>W}RwthA2wS^=B!rA+u@WOTCaIglP6(;U=S}Su z8!e1NeQjn3HX1F1FZlp()5C^XC?N&vWON-1HU@%cN&e4Ad2l_hjSd0)MV81rxzmd2Lb&?D`Bohy&?$NhBjGx{*dp!ZI?huc zss0g(%d^XuND6Xf2tkyLRD+muv0STUL9GS$tgu}c^s2!G=r6wQECMzm`pwxNBqsiY}lm}tr(vY zS(E5sa#1RPP{87my=|{J!-I*#Lip~m;m2tg!o|J;^1VHXTy;^DzV>@jtKf+5Gh|L*o1o9-_Ap6fFAMyF?*EP``;;aR}$E1SYjaL%W(#c_zft z{2n(-Xe%lw%g0*?2(DhjCV>S{)H6c>9~gZu1oRbA*k7=|A?F=l)@t3LkH~(b1`^SJ zwX{)8rM9~a3h?-&ItqpOA3Z7Qq(7{4oulU${vYFWiEQv+%7pc5m}d@Jr@xq!k^#ya zzzXUW1acD9=0tjgKye}~nxwC|@Nvgj!G5#hSaqLl(A7X?ZSQo1-JOJ+<_cz5fE58Fv2FSM+hS@S}iflW>OzBOzE*=Ka2A5i5_+}K8I7fW$nk-33 zl~sKDUzWs~lUv9^=Rc6@@T=RWmrjMYp)yt=!*8%?_{|v}Tm>PDq7^+HSfq6~S}`Pd z$ewy=&pc2PX<}$rH0LHx=2I*HVmI&0Vw&(E(tl0bJjMC!RcCVXkv|sn7Ot3~k%U^6!U6{7O0mJQa zY(g&pjRI@(JY3lXiY%91YOzDXOuix_c348cx`Xb_u7Dxb~TCl^|9Mt8RZBI zO6NY644IQxl;P5km}i`i0yvbo!rDk!`Gj?^FP5x^<*+?NMt87qpU^-!#F-oS2*y)f zN-Gi(FG87oDKF_qB&ioh7g<{757=l-CByw`Yz}`a93|#O32(sdc(mFq))!p{$1$@8 zgM(FM4Mjsl5<3X6X2v3mBeHS=(u3+ZfmOvz&)?+paC-q2-$RYH9)aZiPPEOD&0nF7r2-|;Rd!W*5gX?bbphD4V$sLz zK(?=koaCsd&(^Dm(~E)F#@YLNdR>3z4QSKP={J4&na(_G_MR^OYyA-ha4|_wW?xA* zVUbx|QC`34BJzF5rk>YW#D)^_mznz;^(EA^Bs@Q6C7YF&HVAJD%<#D6P2PRasAV5~$%;PFMm;5cx~B2jf6N0;hU5qCU2-Wd?$*!S8k ze<=c{;mI&;nieA!AV|-PdKQL#D;qBvM~#K2C9lKz-N_LwHx5|mCYe~f4s0M!q+G6hmlZYdYWDpcz^#9DoTyc8X@;tcc2x3QRg&7e5!k!t> zYC-Ht9LX3oS=4&1_?HQOXCh}J=oxVq=>I22pet!m%CPD;noRriRs)XtOgP3I;;E!2 zS|J-te!k*7ku2xt@Jvh)mw;eky}r~J-oF?Td1`uQ237lsOu&WG+CO*J-;2b)8fz5h zNmW7NEh|pgL`~PD0@MkxzD5YPdB#_ZRyijRrC;GT0ws6ABRuCS3QiW!Xqnq+T10F^ zRQ8OZcLNW$@E+dZosJ{1#~8wnnvMJ9@_tp%JjDr}lb&nw^BM2c;V-5%9zIH;xRzNf zzKZ%6`5-$aYisQ=VpD!;PU@rzqG|P-ZfHda?3B-5n5B8ZGIXYh9eEi<`Xef9JW;J zg!%oHE%^5S*LNUbyub^47VsCgQFJ)%#61Xo(*b6256G)+qt9`v1ikWYbH4msgiP{* zC-%?~l>VTFoM)wxz7m>(E>clD8##1T;e!lTQ+mS}AcLco^0uva_4Oz3Y#s#*Y}!{^ z5dqZbAk?fs=OVrov-6i1PZn(Mv8^h;sP&6snkn$9MHtVIxJ=sP4T42ztV|Q6_ z*s~R>M|ycbfd#Go+jM3y7Q=yvU$JJ^6QnEfTSeb$9-Ytq$$;nFf2o!o^8h+=Z77)K zE#xTYHiv;jQvCEK`3&@3Va^)D9zev-tc&NY47)Fh3skG;ep{blcE6!>GI|f-S!Lab z?j(dis7gAZJumSRV!>(-DMVPx0dPtKB9bMiP^oE3Zy|OF~NjR|mx5PH{E`z6Ap^4CwUV2yVy$4853ESeGea80q0^55v1(L?BW}3OHEH> zoHpMZ*KGsaxhH?;$|Q%?K_Pu2LBTy`X6O#V+BI$$Mg7gOujH7ZP=i|0N^-6UKK%H1 zN@kB8|8|BaNS!~!PIZzb!A;wRH@n(OuE6_Ol4Pt&Vci|qE6J3$PEu#!L1Vy}^Gcm_ z;Gw^1KLl?KM!}G_Mn7{Sq&G6(Rit?QwkTndL{EaVnjYVW8ymFGQl7maM;RoH;$=2} zAba(maMwvaVSA7B7>5{C16hiSi|b_|>T*DvsS6?DS8=lhJhoBP!j!ro zm_}ghA~h-rHI}q5DooH(naqiFb3LOoIX*4s13KG71=lJ&^$7&?0X<#*(_<}66qcRI zC{Vz8RC*43hbibSG%zF^G>BmBw3ZRblIsyTl^@{x ziY$)2y>4wJYK0w$_tw0W9o~YnZuSoAR(#KLCMSDj_n*{0pv^f`tYsB?$8`mxr0201 zz<@Cl&-IjF@L{VD##8;?y|FuLmz=3ERp$?B@~n71Zm2mfjm%(s*k5RnVyj6?m^dyS zq+12(_)mnXo6lBF`>{GF`$qhOXz zWwJxHyBN03CXB`LA9|IUZ1ATR4Lx=aiPhe0<9FQsS|Db9PC=MG6p;Ma_=hOuFki{&!8eu0{e{S%OI7;&0pyBDElxeH z1M~P^@{x>o-(}uZFLqYa<@J9_0!l1e3~*Ayaj%+{0kX0gD-l1K7J17_$+u~R0|-$~W&#xv#*nsHF~W9+wW|sAUu52;10! zcAhH`oDWfq&;M2Ul)R4nWoPXden2D9L{v%Qzrj2pE27|Z{crk6y6bV^J^aW|3^sLJ zu$uR80X00x{wbS}#k(!SCe-|zjry&&>XWqU0OOS>1UPG9U|?=c7A$xSTLftvI$bO; zBBwDJo*`?_0?dA4h|T8cg!7#f?yi|hPFm)A-E*#^-SXaoef7Tb&p!Z>L2pSZq7R=} z1fKnRUZ&VcT}+sU%Z>|hqS$)xC`!pvOeO4Ri{V;9Qew!431i6EetD4R?>DN&=eU1d zF%z>C#CvW^RI)kRixd9YqB>5#lk8oeY`*^zpz{L2WCP;^`?XFNDb7$o^(Fkeyywdx zRGWPKQQ(L-_$d)a8z=(-LlYSB>gYj{*V{IKy{n*pNkq(l<>qX+L62f-E6UHSa(!yV zRL@avT=b6lZm+xR__I~^$twE#3ofgA&95IG?kPGI^T76#M8FPu1H!`&X08HZ3VieB)w zp|j!RMo=sKSYSL9-ii8jBK@6!_+RL;cU&mFiOrB}C6l^Cl~p1J*&TjS3e?OnfE3-x zd<1QOsfl@hv+Z`jBFpVFKx2u?(%OwY))Cx8=;qV?{D|&;1-(;9Wy7ikZnQop=q9>1 z7!_FPJlq(G5;~en2kWD{qyCp8P{WY`{&|!7nPSa}`xg|QIdR6_axV`Qtz^0Y3t*vL zhGSDUi;cMj+C}Ciz|qc4g%GZgi+_Rxpr3BD$-HVb^qIJXEMh~y|KY-S0=R?1o>w(IM#M?B4kqt_N^WA zf_ zx6ThvTM&C8S>KQzg09Zk?qI(eW6v!%>k?RsD56t6*kFNMf#M=7fza8Ki)?}7{Ve0s zesRNPOD!r*$+wEvH4FaD27h#Zc*!&}Rk}yB#YHs|8ig%!s$Io=*jCxiH=69;ei?MI z7FNPOxnD}=zBzgC;&I1x**7&I>`xm;DFA@dozV&E%Rs-TiH--#O#`mkTYfPe_iH|V zg|)yi+G^_YU~oY`B?jQY-DfzW)0lTOr7ZkA1R!U$xL`3~?eu9pIDNytj(I_x(s&8S zV}ZyLY1)t|@nQ1nZu=+F4uhn6g*vQbdHkKg-6j}h6N*B>Hab2`DZnuRfbt3b0jcjV zpTo|HTv?`naWS)m-y%&TQrjD6!!oJoa#+g##`T7N+OJcX|7%UtHyz%rrU8T=@fLmw zX0RQ#zsy?1BhO(QRhX`5BxWiui|#u;_~9ZMSa}FJ#L{cI_mq!#f)q538aZf0G5-YPWl+KRc*eM1(t6=aotZ38 zsys1qRPXV+=ki@c!*6ME)bhheq~5C}TlIt*G2dhQ*jR zt&qeZ4ytGK7Je%BN2eXHuxoc(=jmZNnJFoE#m4=)sJqRrHpw=g|!u$ zw7f*=PKX~Py0kb*rz!x!^yGNZ(`@{V`A5IJS z-j&zHhXnY&M5?uX!EkQ#R_oBY{7n;~v7qxbUqpCBqlbA)6?x|zF@a}fxmH>4xZijD zr=S~oQK&sUh0H^aBy`8@iB!tk|B%vNMIjuLCqB*V!x#!X-y^OUO@uoCn*aeF6}5nr z--*z4f!@fN&D03dPj{=X?4j|y8EzK|}yhD`s*rL@vC5 z{nEC2_?&ehtEp(hU&0L}@RgzSP@rNFPlcS7LMb;fQnlQf{{G;PA3rYG07KZf zLSQs{uL5}~LG>1k%BQDg_0t_|o||#Dkf$|hEE(WED@5i&j@PjO3z4mzo$uNcc+q#@ zyX4UhqCc8|-f2Zs0pn%gt#$j~^M99$*ZAmuwxgWA3g#4jmIJlOF-OdZz^bSr6#y4) zWUi{JhfCZHem#D2I(%=?8~|(iQMJeWO81-Zz%*}G)zCO7Pg#S`tPH^ucc#07%J*A4 zTpYL=g5L2w7>(Y!WH$%v5S; z@L=bI+w0T{3yazMgP4a{oLFrHKI`zr2RaToeY?skQ0h=>Io^O6D`DS>iOITQWJcuI z4toFi)~uB9nLUjs{t_NRWAaOtlR6fIg z)-=G$uNapVwt6DwV^_W2B;OrW=N90w6|V>)6)XlB*sL>pMyNFWV&^( zGNMyi`@poMJ?Cc<9G~tzz+efYF4Bv`MNPcLsTY`GCc+^9VQU2@gpMsaN{i(~3#@w_ zV%C%#zJFG`a%yh2Q;n~=q*3AR&;~I8aUdB7nk3U;8D6GaH~*=e5L6M;mMI6lFgM`r z;x*r)E*+(g>bR;0*fyg4zH1Pz7O!r zP2fgUmX5%E*~HvB9)hI1xw7ow$=Rp?ebU9;)Qk|#*8wGGHSKSOh$IcH=MY)ws`l8PsF$g6=(EUTYpgUV4MHIUKkPYBeGO_T? zp|f4v+CEJlLaTveT3FUe0P{rpB&x2ierOf6F-A^KE@Y3i6Z8H$jXisBn2*Uc)$)}L z_#<<~CNHzA(Wh2w72F?;cHF|EPyDq_oh(mg9^L8N+)s&q`}ezU)3m^x`CbN@nOIgi z?kISH(k_RQD7Q$<;aHb5w7N&N8riK+{J7O}`a^mD5#+}XSLQ)q!>pLnjwis6XS_ou zg=t0(4n{Et&Ihs$n#hwN7G3`veFI@S=w$K*62{Ui zFs=C^3X0-o@Y-9lc5t$fK93o}p|>|~*2AfHez=rOr|?K`yPH*T#5-Om;wgkss5nc# z+aA~SQ_1Fgt78{We9h|9V205W4%#u_X2hG$0J+#kdL|RnDEOP1HPoW7sY01Njo9*^ z@c)}He>s|)b34puv%j;4iH$GWVdwa+cCzZuEs_6wr(+-?$SDII(?TYKrnZt2nk;V* z0ptl2@|KnjOowC2C|E;7OME^m?SLr5jel=hZQM2fNpSLUZF@?M;|RR%kSwc@j~(Xd z^PV|_nSqJ%z!-a;CuP>t9&D1p`sRy+7Jj2TA*y8+7Wij;ex+4ns-aahP!5~3K#KI*Yk!?wh>Jq+_sVc*MaHcW>G4<-Vz$;X#R4pp24-w zge%mJ1jEPUK%MlqQON!Cn_(o~gB8@rC7EvZ34g{RwCF`f>n{hh6$72(NM!_d?C=Tk zWJ53<3K0pjuZYVEc`^!mQ9Y^$?BbGujihAY7D78 zU&=V}w*7Soz7qxH!d?z45TwI_!Fj(z->?KuX+G!r*Z50Bk5Z|~L^m$O9u|4c2CG{J zg4g89)5>7*0Yn`g@QGyZRGk9Flx0=6m7YgUO-`SqMW9!RMcyqpITRD?^nqbQieLLO zSS-REqL|ir%a_cOki+D$tLiSzVOM2{FLv;ny~yp4;N*baa;*#vD7m~dI+(X->d#zM zf?PsM1Vi{qbnh%f#|ZR9@x(V++$Y=z?Z4Y4lBTRT2X5q23WwDP*eh4zR2!VEl(baD z9In|H>6Yr>w>!0}txbM(x%{F2%2<32iw|s(n+)Xy!4mxsSfUeVe$of~lt&~!x@Tnx zWPHr3Mn0J$?=(JFEuMHtFFtAclm zg9qt)?{5AU!v8w6n)t2z`;&mrDW9|fpX7YCl*V}#Lqud`QOU{4`#B;$JIt&_&UnQ0 zKSRGx=WyPv9EM>XUtCzcg{VsqD?92Q=|8#BInyG4yTB^af**{1`@Ofz2K7@uCN1?j zV`i?i90*{&2n|;_QMJud8i_HBbhcl_lmA-`*6rQ6KemXyWhs^G1N-enouXq5TO_N9 zlT=gQa}-48G&zqowU-rM$QVCULEV}KoFhcpn>XgePh7Ya*XwhbY-#2rU7 zj19S{^xA(0r9EF*VMb+k5_6sCPJ{~gE9H*;t!SIT@@-W*X8X6&gRGpMeOz` z`JOtpirRSz%o;)#ZY^aMoXl~w;TKb!w;8SL`+N*;gSGwko6C!Ssu^2V%Red%0WuL-i|F-HvUHsuF{bgO?kzZ^1MT)1HVqv<3U zLc@;(0VbrfQkyG*H#y#as6lxl(Rt0_j*o~~y02X15PP6l-&w`vF;U?uJH z^ytI%<_k+ki&`CFYXTUvJvy%fhLPD?fsfK?+B>;7%rv=0&i30yBMx3jgZy!uYf(t% zVTTNUJ&kyz_|mey_Ue22o~0h*s!aRa;%sd8#o>Dwzxf25c8HsNx)Q1l@9;Z+G^?sc z>deh8jXi>J?=r*r|M($K<9)eR3A?CE$z=J&0-<+umeJD)Lq!7KfN5%t{{yWQ&UI&;bRu$EMU z!_UA9@b`7F@1D67nqfP!BEngRvZt_<#)qrJt?ZfP2XBK0^SSukRCeR36=O`GR&sUC z$vc-?E^hh_CJ5nIgjDxubbxg^_ud7v8-=r`zNvl2g2dh7pgr@HUzkI6d(xc z6xPmSHqc?fpuGgz$bH2%!2B}<>Q((^dYco(tl8etk~?uT;QozUW{Ok*zW)TWhdLai z&iuiQt38`@$u^?1w4-mrd63xbnM5mg2C8+u4z@`7B%c4Lu`dsYvW@p=j9rYi$THSY zgzUnMecvf-6dK#BtSN+vL?la=>^p@rlxRVbJzKJeWG_TQmh+w7_nh~f^SiF!b><(} z^d%{l)S!@SY~rO|B8%I=z5U?akSC=2r5a z=Cu;ZKr>a15OQ+z^?hU1C4lfPOTbxzf1#Rdto*=U=u;>Aq(;A25(pnohMI zas9Z4+j0F|>v5EsQeP|9XjF<#6Gc!}?SQ;I`*!$zfCvo?9xih%9`%;K`tOn`|1#<01hKRY)b^gA@H;+XV z>|X36TzuLyE>bB;pM6thulC5QQsN=Qy&<06KQ&-gYds?K*B$&KGA8&_fe7NCHoU#F z(-(X6wEQkj-$k@;TlF49aWRcH-BCcJ(e`5Zi-!33%hfT<^9~cDg_c^3SNuD*o(5hL zhIOC)n_lHnN-6cp0OmZ$GF8I-`S<52ksGS z8g3zS*U>4W))7L+8EJXPExC7HmDKz1bMd0*C*x&1Bh1DAYqRJ>Yh+o^hU|(Uv6E2w zwsmt9t)ycj|M%`rD??Ct>c6&jU<%tI&_Fo5+3gcr$gHG}>3Nmql@?iS`_FFAf3d$s zjbG<^MA$PF#;rhP!^#c6Lc)=pTgS970PGZAPhqjAj&n$wBhUfriID zcBRoe=*QBRP2HppyCe2-XX|*hZRIKoCa@1ht_7lYhANC}J${>f)k|p`8+!;Sj1C-Q z<^-;DdK|VM*Hvq{uTDX|fpR~R4-6JEkdQH5@F*wpgg1H}JukhRn8@nMR8W7jzjgdD zHsWD%2Sgtjs_w0k*}o1>-w@y6Anc5yN>((ls~iT3~ik$scvP*6^rqnEA=nXbK6}lyfyUy-90HL z7HEv|2XhoLgLJKjcZiyrTH0$=dKb>z^12R{b}tM@lYuQ8-L3&@ktDl&CSJ2IbPNpy zl|+8hRuh7Ee0s(m_}@`Ry%-XZ;Sua!f>S0U^zCZK(y@9NR8Q!y8Ixyz;W zfEWajaW+Vbxdehfu2^2dxx0TX*M!3D7aMc53flG6_3H@$kNv zLB6{ygbxcdEwPy@BpZk^t*`#Gi=<6Jjg`LCpRY+fq;n_jC}!g=QTG-C>{wS%lGU1tTL70cIIN`bBeEaf-9R0xA=iPkNmhM2eQ+LeX{ILXxq z&vO$OV8Y6&x(7s8bB<`5w&nPJVExx$Zof&biz4T-&N?-3ei+Zy1L#@+V*u$-%5FpHdJlT;q1Q zx502}9Ws+7^WGlgo72R!qiHUR&WuyhWxr0XuPxP)>71ROXZn7X5!=Yf*tja>uEec; z^Y#NTg$ny>vbDVbz84-4W$SKd*C0je7zGEPO^QfH+VCXM3`R%fMhk_`=7p1tl-WKP z_fSK(>5W5FhAhKk5?2iww#gs0-RTLDyQ=sw0eEc5)W&dER1W~Nb&5kFH5Kw$!?&my zMAK&Ix?ON330y3@F9Gvl=1X8honhg&Y@d;s84b!o`kD1yg`{ z)TrI#QJo9IV4}kWYgEW-SR6Ak2k15XW;>!4AMDFGna{Q)QyQ{MJM}I(a)HD{-~0FM z4Q7;yRJ2OmO?#ZxyCBA6DmQq3{$+l~-+=+{>}JN$f9@_dME(<#}3wNV;q#SBOo*k6fv08SO0etnVD5isWS`(KCL7~%`R z+h{;eNX`R1XpTQY3g?9xm-U|PDwc^{347UimqHE66k&6|R}Cx;@BP00O!MlH)qz$! zbpkjV^*nW@3@OlJYwb!Dr1eN(nfFXi4NIphq%1V}>~{Q?8$c+EzGILj(LbiLGl@hd z)8z-Qj!R9uzj${awS9e>*L|$+f1W;46qtmn+1{&@jCxiD()7P{3AY$*IO{dcSR7J~ z_rfF(W!N7TX?S>Y$4cY@FTpBl*V3O|BhNZ4-33fWbpwvuBoRg5Z}b38Sp){*)T-(F zan%JhRciuQh#9h*-aubsAG5c=pMK3COHoF%nk=960y-^^t|j5_!c8zh-;1R>laLL< zX;U2?s;YCmO1-m=A<8$lW;-Rh+e~bnAx)x?w)nI>Gky&_hJQLW51#%Gq8DDD{Y6PQ z##uQ1iHwOVDLtlSDmD?G=>#(1;32(p&R?T`Z*^gYN`Zr zcobC$rRsj7!isMsoL`~~)f*O;8^-CsG6xGi&2*<2U%LmyP!^&+_}-DFR;7kqM4W{2qy*wrlb3@G222y^@edZK&NRS*hX1ZOjlTYJTc8FV^Gll2;2|%G6alXJ^*<2omWLLH!5$ zaZ2$Cfn`2^2RS!Igsd*u0(d=}@5mY1OnP(ZO@nr#r*CyJ)=F_QPvksc60v zJ-{FtVW7N4q55OZI~*_0shEO0RoXHYCC!?H#m%yAjic?zDGfrw z?smqYNjsY3$2?zBWoj&-0NYrt7L}5E%M0_9YSh_jr8;-Id@{uXc2m%Z&VfhBBRey- zEpt+I@Nq ztGde2bxtCQx1l!Li#Xd*^cZJcY?3A1H}$&H80ql~HxT|QsEftzb93tM)?&LGuLhFz zh8D?#h_Fc2fht=yzs}*hWtI?&s?z;LE*HW7+u3ZfGfDf@dqY^w1T6*F&_4c{{?MisxX;&k}5Tfo`EkXK^*5_xH-DhltbhT++4GWdA;Vd;? zj|7f)9ed|f%gTu9d0E`8k{e~Xzi>$L`viN?jkch2JHDeIAI*LZcGi}PwpnNCFp<@4 zl)x{X>H}T4YK@eE9lvhTCbtWE z#E6{Do!$xg#$;3L6S=+CrrUoCvDUXZ_=2`}v6nwxi!w=tEi}tP=xjlcq230^q>hDN zm`8x_TvnPhO6GKS|3g_u9TEJ(uaI{;77rU-_N{p{{6nYQmcIY0r;E#L{Bt~FL7~v5 zHbbAe*(JGpfoE>6vX`~ymN8L?9L-;0w+SkHg2yuN8BN%&|9dhPt%?~ zo~oi}pk#e;A75Qxm1#uaXV(x5L=|23LMPZznC>6HaQsY)^)M5u04}R;N#5VflE0hj zEF4D92GYcFDMit86mVxpqa6*Axn){K`1LPuiX6za1U5eO<#ZOD_<-BtdEfTN+`l77 zsbM3L3x-=azv!hAMEIaowCuWwXY68I8iy8%;I&E(Rb94p7kQWD2j?Yx5s^{IfhjE6 zizkv&b|5z9^no=2lIQDPff?jXSou4#1EX_ z=Vw~z3cCCv+WG3Ao6f;U_(%pq+xd20Y#Loh3=ECBkk*9$G;{y-nk;1e*^$fIAArMm zJ+#p=&`b8Jni;9Vemy&v1~zpvK#4=9du)ucxRQr`ET z*tKYQ#$Z`#ap&c$S7nY(5y?;`^YQg%CZuv;vK}AlGFx9?w~Jke4Ld+HSf>)BzUjBx z!n0GM?rRR_*Q*w9Z#HRvTc~(GVVQ?tnp=K;yRXgo(Q;;DmJI>lqEBIJd0Crdgc0E` z6+PPL<6Lf~h2Hd(_M#55KSL`RaQFWFhwQ$mz59N1B(EJ`?Pm&_kLZq34OFXlqN=;wTk zM|K??dL;uZQ+_a+mArb-=`-zqCR%ZJ#>LcCo#f!|d-3r+t0NsJ^ks3+u!a2C#x5#! z&=gP8v_*j*w_fMzJ`rL^mf3Ji`tBJOCdsbhC9BIm{1*xsc;_`v$yA*i!X3Fcamq|+ zNrDPfu<~6iD*Qt_^Dh3csiW%1u5_$LFDueOljL;N|6b)s2EWfLVlk|U=$eT3rOhH?QETC&GF6uHy3|yb zS$og#4#rBm{5E5SGw>A6jui0AE|bwE4fEkk zr%qF)(UKOf%7;#0P-JIRKU&pl))ee*?r<+Nc}!X9#k`-(A@`?K>5<5E+AEYRl!bGP z8%L3|8N@ygG^r*rAzgO9PU^KT^dC`<{NCl)mX z5zpGt_?jL4HUBgG; zx$^9B`eFsAweY+E9cgen&K9Uf`mlj;)U1mZeNwH|6cU56{f-}l+b-V4?kQN%I z;0on)?8flENYV-*i)*(SLC0f7>oUjkC-l+fv*qN{p{b~#Sz4fTD)6}k9a=B9Fx?tL zf#A*<87MLn-?=fKsKuFeg*Nk(Ce3N_JC@PqYZ+^Z6Wub8Q_ylRK!R4X>dnvt&kxzkgD-q9 z*9UDByWid0+gkvneQsH!CLFlnUMu)6wLh)HHg-@Et*F2XOAD~fnfveW@fZ_1%20MT z15w`&fhf|0aJ&UjHJi#b0AJG-cwYuWN_7WNpRka5AZB2cDGqM2Q?*k%%2-Jnf$H#GCu| z3?N{+)^KC~!+aB!$55Tw3JudfG;qdTzNYg&{ zc>&NH*`V(LeQhpRy;+*f9t@Da)Jmu5J4dH_Rs)_p1|=bfXb4;_hGKCqIAjZ}sAR{m zh=3a!7vwl|^Z|)y0i=kt)6-2ESwOfl1>VUbfONCpzI8h<01#XZGz0gsXJY$cms11A z9zxPF?2lZ#GX`ti%tt}tn{xU>cC8MCB-a3<7W!I^z;f(Z+x}cn*28)5GN~p;_S9eh zDf+c5J*igDIQwiKR1_W_N0clYx5W>5`tQ<)0A1%})rB0N*)}ZroqRpcAt(D+BQQ<| zKtYredP1aT8$Nm`j@(;*H)37Nr6bur2UJ24w&KJAl|ps)=0PS9jx@mnBOhpNx7WIM zz~!OLxjk{^kuO9~a{wT0$hPZ zw|9ubxdLZR9c-y)-t@_3KL_XapF8@z-UuK5=ACP>yK7@69S!QhaJ%2BtE zlcOuOfXb)=*hmdj9AQAb9I5dfNhHRVdV#CGdSzwhCef-FFraOx-U?f}zX&%ELj%?x zD<ObZcG9cXd+vb<_F~;s-M2H0Qw@M%Pg06b!TY)KglhUI9e+L%GX5$8o0P6BBO0 zt%2nE)3R>iD~--FE*&gF-18rDE}N(Df$n<#GLH_uv^@CR4BgL~nsU@LFerpR^5YB& zaS<ghak9}ZkTtYZbmYTHoIS>Qt!wm&6Bz{a zq|7rAnA&=v(dRef=DBs4kYMb?0Kyy{kY%hCkVTMrDP-T??lHjx^wN86k~`Ces$E+g ze;Gt5enKc|2e*WhiRv}GysKE@Ib+rRBC32QswIMw9sLdJxL<9;h6&0Z14W##G26r0RyiVXsa`f$reZ4(H`}*m+RZ=3R zWfbCUg(PBXaD5RxEKkmSDQ+brji8MF@tk+0$u=>Oz$f-mqDDh4fOF$?H=etOv`(%nhR-&v4 zY^nF&$8tip*gXPrUzz^EG&0P?B;;yaZ37<6;?TI)%+;4hlv5|kiSrQDs&1{wzZHLE z$3H0qkt)ie9^obw@{24Gk7*1;~1h zzAKiTO%Kbt7urx`@pgt#;Pa(0{qGeClL7L#`(%ko(%F=-Vu~f3S$x>&PnZMdbxuYz z#YL{70u&%O`SVV}eC0)!0R}ZBjx%j~=`7qY5)v|dT@9>7$v+F7 z0&&HpXDMR%XJLRJy{BNnOE<_!N&b6Hf|6J(2+!Z-{dYagAZo5%pKA#T{d;YWC__^) zNmltYEB{%PF$8FN-^x(iyMM1Wa>6x9${705_i%&Dbd!XV1@+k+oM2K?F#kam-`m&U zic(r=?X3RuPgf*}w{VRC*=8FdQ}Lho#R3AGKbv9tY7Vv*)7zKa1r>c<-mj zYPLQ3)1*TXAV?fdJyeMiy>2Ib)G z1lj?S+D9fD41$6X|D#6~w!*4208Gf88-oLCrpC@j4h;cfNn9+=>g;z;zu8Od)R}F& zkB?mpM<3xnd%hsLt>&M5w897r!0An>p3>ci5A$oMV`m%_YYHMi;mHAXw zEsj90a>(y*&pn@Y024^et}emQ3C3jlKK{hXzdISKmyXzL39O?Jqiw1Q*VZq_*U`|^ z3o$S-2q_XxmI#AT88H)7I=>F03>_jNd*fW0S&@8mXJ@DW4ac^-|9R)xph*D@{!&-} z?foWr*geSz>mL92(i28 DxnI)$ literal 0 HcmV?d00001 diff --git a/topics/kubernetes/images/service_exercise.png b/topics/kubernetes/images/service_exercise.png new file mode 100644 index 0000000000000000000000000000000000000000..b924eb6930f3644cdabb2bdd224f16cdc92a34dd GIT binary patch literal 27380 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 A(f|Me literal 0 HcmV?d00001 diff --git a/topics/kubernetes/images/service_solution.png b/topics/kubernetes/images/service_solution.png new file mode 100644 index 0000000000000000000000000000000000000000..5e01c4c57edc56132f8f3780eeb8ac2fd712a938 GIT binary patch literal 25341 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 rs.yaml < +``` + +4. If you'll list all the Pods now, what will you see? + +``` +The same number of Pods. Since we defined 2 replicas, the ReplicaSet will make sure to create another Pod that will replace the one you've deleted. +``` + +5. Remove the ReplicaSet you've created + +``` +kubectl delete -f rs.yaml +``` + +6. Verify you've deleted the ReplicaSet + +``` +kubectl get rs +# OR a more specific way: kubectl get -f rs.yaml +``` diff --git a/topics/kubernetes/solutions/replicaset_02_solution.md b/topics/kubernetes/solutions/replicaset_02_solution.md new file mode 100644 index 0000000..1f73874 --- /dev/null +++ b/topics/kubernetes/solutions/replicaset_02_solution.md @@ -0,0 +1,62 @@ +## ReplicaSet 02 - Solution + +1. Create a ReplicaSet with 2 replicas. The app can be anything. + +``` +cat >> rs.yaml <> rs.yaml < running_pods.txt +``` + +4. Remove the label (type=web) from one of the Pods created by the ReplicaSet + +``` +kubectl label pod type- +``` + +5. List the Pods running. Are there more Pods running after removing the label? Why? + +``` +Yes, there is an additional Pod running because once the label (used as a matching selector) was removed, the Pod became independent meaning, it's not controlled by the ReplicaSet anymore and the ReplicaSet was missing replicas based on its definition so, it created a new Pod. +``` + +6. Verify the ReplicaSet indeed created a new Pod + +``` +kubectl describe rs web +``` diff --git a/topics/kubernetes/solutions/services_01_solution.md b/topics/kubernetes/solutions/services_01_solution.md new file mode 100644 index 0000000..da60267 --- /dev/null +++ b/topics/kubernetes/solutions/services_01_solution.md @@ -0,0 +1,19 @@ +## Services 01 - Solution + +``` +kubectl run nginx --image=nginx --restart=Never --port=80 --labels="app=dev-nginx" + +cat << EOF > nginx-service.yaml +apiVersion: v1 +kind: Service +metadata: + name: nginx-service +spec: + selector: + app: dev-nginx + ports: + - protocol: TCP + port: 80 + targetPort: 9372 +EOF +```