赞
踩
Kubernetes 中, 容器总是以 Pod(容器组)的方式进行调度与运行。因此对 Pod 的理解与掌握是学习 Kubernetes 的基础。
Pod(容器组)是 Kubernetes 中最小的调度单元,每一个Pod都是某个应用程序的一个运行实例。以前我们的 Web 应用都是以 Tomcat 等 Web 容器进程的形式运行在操作系统中,在 Kubernetes 中,我们需要将 Web 应用打成镜像,以容器的方式运行在 Pod 中。
Kubernetes 不会直接管理容器,而是通过 Pod 来管理。一个Pod包含如下内容:
Pod 中的容器可包括两种类型:
在 Kubernetes 中,我们一般不直接创建 Pod,而是通过控制器来调度管理(Deployment,StatefulSet,DaemonSet 等),这里为了便于了解,先通过 yaml 配置文件的方式定义 Pod 来直接创建 Pod。定义配置文件 pod-test.yaml 如下,
apiVersion: v1kind: Podmetadata: name: pod-test # pod 名称 namespace: default # pod 创建的 namespacespec: containers: # pod 中容器定义 - name: nginx image: nginx imagePullPolicy: IfNotPresent ports: - containerPort: 80 hostPort: 8081 volumeMounts: - name: workdir mountPath: /usr/share/nginx/html restartPolicy: OnFailure # 重启策略 volumes: # 数据卷定义 - name: workdir hostPath: path: /tmp type: Directory
其中 spec 部分的 containers 定义了该 Pod 中运行的容器,从 containers 的复数形式也可以看出一个 Pod 中是可以运行多个容器的。
执行 kubectl create
或 kubectl apply
命令创建 Pod,
[root@kmaster test]# kubectl create -f pod-test.yaml或[root@kmaster test]# kubectl apply -f pod-test.yaml
该 Pod 创建后将会拉取一个最新的 nginx 镜像,运行一个 nginx 容器,并将容器的 80 端口映射到宿主机的 8081 端口。
可使用 kubectl get pods
命令查看当前 namesapce 下的所有 Pod,加 Pod 名称查看具体某个 Pod。 如果需要查看 Pod 调度到了哪个节点,可加 -o wide
选项,如果查看 yaml 文件信息则可加 -o yaml
选项, 如下所示
[root@kmaster test]# kubectl get podsNAME READY STATUS RESTARTS AGEpod-test 1/1 Running 0 116s[root@kmaster test]# kubectl get pods pod-test -o wideNAME READY STATUS RESTARTS AGE IP NODE NOMINATED NODE READINESS GATESpod-test 1/1 Running 0 2m19s 10.244.1.42 knode2 [root@kmaster test]# kubectl get pods pod-test -o yaml
如果要查看更多的信息,可使用 kubectl describe
命令,
[root@kmaster test]# kubectl describe pod pod-test
该命令输出内容如下图,
各部分说明:
当 Pod 在调度或运行中出现问题时,我们都可以使用 kubectl describe
命令来进行排查,通过其中的状态及事件来判断问题产生的可能原因。
通过 kubectl exec
命令可进入 Pod, 类似于 docker exec
, 如
# 如果 Pod 中只有一个容器[root@kmaster test]# kubectl exec -it pod-test bashroot@pod-test:/## 如果 Pod 中有多个容器kubectl exec -it pod-name -c container-name /bin/bash
如果一个 Pod 中有多个容器,则需要通过 -c
指定进入哪个容器。
Kubernetes 对 Pod 的更新做了限制,除了更改 Pod 中容器(包括工作容器与初始化容器)的镜像,以及 activeDeadlineSeconds (对 Job 类型的 Pod 定义失败重试的最大时间), tolerations (Pod 对污点的容忍),修改其它部分将不会产生作用,如我们可以尝试在前面 Pod 定义文档 pod-test.yaml 中将宿主机端口 8081 改为 8082,重新执行 kubectl apply
, 将提示如下错误,
[root@kmaster test]# kubectl apply -f pod-test.yamlThe Pod "pod-test" is invalid: spec: Forbidden: pod updates may not change fields other than `spec.containers[*].image`, `spec.initContainers[*].image`, `spec.activeDeadlineSeconds` or `spec.tolerations` (only additions to existing tolerations)
通过 kubectl delete
命令可删除一个 Pod
[root@kmaster test]# kubectl delete pod pod-test
在 Kubernetes 中,一般不直接创建,更新或删除单个 Pod,而是通过 Kubernetes 的 Controller(控制器)来管理 Pod,包括 ReplicSet(一般也不直接用,推荐Deployment方式), Deployment,StatefulSet,DaemonSet 等。
控制器提供如下功能:
Pod状态并不是容器的状态,容器的状态一般包括:
Waiting: 容器的初始状态,处于 Waiting 状态的容器,表示仍然有对应的操作在执行,例如:拉取镜像、应用 Secrets等Running: 容器处于正常运行的状态Terminated: 容器处于结束运行的状态
而Pod的状态一般包括:
状态之间的变迁关系如图
Pod 刚开始处于 Pending 的状态,接下来可能会转换到 Running,也可能转换到 Unknown,甚至可能转换到 Failed。然后,当 Running 执行了一段时间之后,它可以转换到类似像 Successded 或者是 Failed。 当出现 Unknown 这个状态时,可能由于一些状态的恢复,它会重新恢复到 Running 或者 Successded 或者是 Failed。
定义 Pod 或工作负载时,可以指定 restartPolicy,可选的值有:
restartPolicy 作用于 Pod 中的所有容器。kubelete 将在五分钟内,按照递延的时间间隔(10s, 20s, 40s ...)尝试重启已退出的容器,并在十分钟后再次启动这个循环,直到容器成功启动,或者 Pod 被删除。在控制器 Deployment/StatefulSet/DaemonSet 中,只支持 Always 这一个选项,不支持 OnFailure 和 Never 选项。
提高应用服务的可用性与稳定性,一般可从两个方面来进行:
Kubernetes 中对 Pod 的健康检查提供了两种方式:
Liveness probe 适用场景是支持那些可以重新拉起的应用,而 Readiness probe 主要应对的是启动之后无法立即对外提供服务的应用。
就绪探测、存活探测目前支持三种不同的探测方式:
以 httpGet 为例,示例配置文件如下,
apiVersion: v1kind: Podmetadata: name: pod-testspec: containers: - # ... 与前同 - name: workdir mountPath: /usr/share/nginx/html livenessProbe: httpGet: path: / port: 80 httpHeaders: # 此处header无意义,仅作示例 - name: purpose value: for-test initialDelaySeconds: 2 periodSeconds: 5 # ... 与前同
删除之前的 Pod, 重新创建,使用 kubectl describe
查看,可看到 Events 部分如下图,
Http 存活探测失败,状态码返回 403, 导致容器重启。出现这个错误的原因是前面做目录挂载时将 nginx 的 html 目录挂载到了宿主机的 /tmp 目录, 而 /tmp 目录没有 index.html 文件,导致请求返回403, 在 Pod 调度到的宿主机 /tmp 目录下创建 index.html 文件即可。
echo '
Hello, K8s!
' > /tmp/index.html
其它 Exec,tcpSocket 探测的配置示例如下(配置在 containers 元素下),
# execlivenessProbe: exec: command: - cat - /tmp/healthy initialDelaySeconds: 5 periodSeconds: 5# tcpSocketlivenessProbe: tcpSocket: port: 8080 initialDelaySeconds: 10 periodSeconds: 10
支持的参数说明:
readinessProbe 配置与 livenessProbe 类似。阿里云上配置就绪检查如图所示:
健康检查的结果分为三种:
健康检查的一些实践建议:
本文对 Pod 的概念与基本的管理操作,Pod 的状态变迁机制与重启策略进行了介绍,对 Pod 的健康检查进行了详细的了解。但在 Kubernetes 中,我们一般不直接创建 Pod,而是通过控制器,如Deployment,StatefulSet,DaemonSet, 因为控制器能为我们提供水平伸缩,rollout(版本更新),self-healing(故障恢复)等能力。我们将在接下来的文章了解控制器。
作者:雨歌欢迎关注作者公众号:半路雨歌,查看更多技术干货文章
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。