Kubernetes基础:Pod的详细介绍

1. 基本概念

1.1 Pod是什么

Pod是Kubernetes中能够创建和部署的最小单元,是Kubernetes集群中的一个应用实例,总是部署在同一个节点Node上。Pod中包含了一个或多个容器,还包括了存储、网络等各个容器共享的资源。Pod支持多种容器环境,Docker则是最流行的容器环境。

单容器Pod,最常见的应用方式。

多容器Pod,对于多容器Pod,Kubernetes会保证所有的容器都在同一台物理主机或虚拟主机中运行。多容器Pod是相对高阶的使用方式,除非应用耦合特别严重,一般不推荐使用这种方式。一个Pod内的容器共享IP地址和端口范围,容器之间可以通过 localhost 互相访问。

Pod并不提供保证正常运行的能力,因为可能遭受Node节点的物理故障、网络分区等等的影响,整体的高可用是Kubernetes集群通过在集群内调度Node来实现的。通常情况下我们不要直接创建Pod,一般都是通过Controller来进行管理,但是了解Pod对于我们熟悉控制器非常有好处。

1.2 Pod带来的好处

Pod带来的好处

Pod做为一个可以独立运行的服务单元,简化了应用部署的难度,以更高的抽象层次为应用部署管提供了极大的方便。

Pod做为最小的应用实例可以独立运行,因此可以方便的进行部署、水平扩展和收缩、方便进行调度管理与资源的分配。

Pod中的容器共享相同的数据和网络地址空间,Pod之间也进行了统一的资源管理与分配。

1.3 常用Pod管理命令

Pod的配置信息中有几个重要部分,apiVersion、kind、metadata、spec以及status。其中apiVersion和kind是比较固定的,status是运行时的状态,所以最重要的就是metadata和spec两个部分。

先来看一个典型的配置文件,命名为 first-pod.yml

apiVersion:v1kind:Podmetadata:  name:first-pod  labels:    app:bash    tir:backendspec:  containers:  - name:bash-container    image:docker.io/busybox    command:['sh','-c','echo Hello Kubernetes! && sleep 3600']

在编写配置文件时,可以通过API Reference来参考,也可以通过命令查看。

[root@devops-101 ~]# kubectl explain podKIND:    PodVERSION:  v1DESCRIPTION:    Pod is a collection of containers that can run on a host. This resource is    created by clients and scheduled onto hosts.FIELDS:  apiVersion    APIVersion defines the versioned schema of this representation of an    object. Servers should convert recognized schemas to the latest internal    value, and may reject unrecognized values. More info:    https://git.k8s.io/community/contributors/devel/api-conventions.md#resourceskind    Kind is a string value representing the REST resource this object    represents. Servers may infer this from the endpoint the client submits    requests to. Cannot be updated. In CamelCase. More info:    https://git.k8s.io/community/contributors/devel/api-conventions.md#types-kindsmetadata    Standard object's metadata. More info:

    https://git.k8s.io/community/contributors/devel/api-conventions.md#metadata

  spec <Object>

    Specification of the desired behavior of the pod. More info:

    https://git.k8s.io/community/contributors/devel/api-conventions.md#spec-and-status

  status <Object>

    Most recently observed status of the pod. This data may not be up to date.

    Populated by the system. Read-only. More info:

    https://git.k8s.io/community/contributors/devel/api-conventions.md#spec-and-status

[root@devops-101 ~]# kubectl explain pod.spec

KIND:    Pod

VERSION:  v1

RESOURCE: spec <Object>

DESCRIPTION:

    Specification of the desired behavior of the pod. More info:

    https://git.k8s.io/community/contributors/devel/api-conventions.md#spec-and-status

    PodSpec is a description of a pod.

FIELDS:

  activeDeadlineSeconds <integer>

    Optional duration in seconds the pod may be active on the node relative to

    StartTime before the system will actively try to mark it failed and kill

    associated containers. Value must be a positive integer.

  affinity <Object>

    If specified, the pod's scheduling constraints  automountServiceAccountToken    AutomountServiceAccountToken indicates whether a service account token    should be automatically mounted.

1.3.1 创建

利用kubectl命令行管理工具,我们可以直接在命令行通过配置文件创建。如果安装了Dashboard图形管理界面,还可以通过图形界面创建Pod。因为最终Pod的创建都是落在命令上的,这里只介绍如何使用kubectl管理工具来创建。

使用配置文件的方式创建Pod。

$ kubectl create -f first-pod.yml

1.3.2 查看配置

如果想了解一个正在运行的Pod的配置,可以通过以下命令获取。

[root@devops-101 ~]# kubectl get pod first-pod -o yamlapiVersion: v1kind: Podmetadata:  creationTimestamp: 2018-08-08T01:45:16Z  labels:    app: bash  name: first-pod  namespace: default  resourceVersion:"184988"selfLink: /api/v1/namespaces/default/pods/first-pod  uid: b2d3d2b7-9aac-11e8-84f4-080027b7c4e9spec:  containers:  -command:    - sh    - -c    -echoHello Kubernetes! && sleep 3600    image: docker.io/busybox    imagePullPolicy: Always    name: bash-container    resources: {}    terminationMessagePath: /dev/termination-log    terminationMessagePolicy: File    volumeMounts:    - mountPath: /var/run/secrets/kubernetes.io/serviceaccount      name: default-token-trvqv      readOnly:truednsPolicy: ClusterFirst  nodeName: devops-102  restartPolicy: Always  schedulerName: default-scheduler  securityContext: {}  serviceAccount: default  serviceAccountName: default  terminationGracePeriodSeconds: 30  tolerations:  - effect: NoExecute    key: node.kubernetes.io/not-ready    operator: Exists    tolerationSeconds: 300  - effect: NoExecute    key: node.kubernetes.io/unreachable    operator: Exists    tolerationSeconds: 300  volumes:  - name: default-token-trvqv    secret:      defaultMode: 420      secretName: default-token-trvqvstatus:  conditions:  - lastProbeTime: null    lastTransitionTime: 2018-08-08T01:45:16Z    status:"True"type: Initialized  - lastProbeTime: null    lastTransitionTime: 2018-08-08T01:45:16Z    message:'containers with unready status: [bash-container]'reason: ContainersNotReady    status:"False"type: Ready  - lastProbeTime: null    lastTransitionTime: null    message:'containers with unready status: [bash-container]'reason: ContainersNotReady    status:"False"type: ContainersReady  - lastProbeTime: null    lastTransitionTime: 2018-08-08T01:45:16Z    status:"True"type: PodScheduled  containerStatuses:  - image: docker.io/busybox    imageID:""lastState: {}    name: bash-container    ready:falserestartCount: 0    state:      waiting:        reason: ContainerCreating  hostIP: 192.168.0.102  phase: Pending  qosClass: BestEffort  startTime: 2018-08-08T01:45:16Z

1.3.3 查看日志

可以查看命令行标准输出的日志。

[root@devops-101 ~]# kubectl logs first-podHello Kubernetes!

如果Pod中有多个容器,查看特定容器的日志需要指定容器名称kubectl logs pod-name -c container-name。

1.3.4 标签管理

标签是Kubernetes管理Pod的重要依据,我们可以在Pod yaml文件中 metadata 中指定,也可以通过命令行进行管理。

显示Pod的标签

[root@devops-101 ~]# kubectl get pods --show-labelsNAME        READY    STATUS    RESTARTS  AGE      LABELSfirst-pod  1/1      Running  0          15m      app=bash

使用 second-pod.yml 我们再创建一个包含两个标签的Pod。

[root@devops-101 ~]# kubectl create -f first-pod.yml pod/second-pod created[root@devops-101 ~]# kubectl get pods --show-labelsNAME        READY    STATUS              RESTARTS  AGE      LABELSfirst-pod    1/1      Running            0          17m      app=bashsecond-pod  0/1      ContainerCreating  0          20s      app=bash,tir=backend

根据标签来查询Pod。

[root@devops-101 ~]# kubectl get pods -l tir=backend --show-labelsNAME        READY    STATUS    RESTARTS  AGE      LABELSsecond-pod  1/1      Running  0          1m        app=bash,tir=backend

增加标签

[root@devops-101 ~]# kubectl label pod first-pod tir=frontendpod/first-pod labeled[root@devops-101 ~]# kubectl get pods --show-labelsNAME        READY    STATUS    RESTARTS  AGE      LABELSfirst-pod    1/1      Running  0          24m      app=bash,tir=frontendsecond-pod  1/1      Running  0          7m        app=bash,tir=backend

修改标签

[root@devops-101 ~]# kubectl label pod first-pod tir=unkonwn --overwritepod/first-pod labeled[root@devops-101 ~]# kubectl get pods --show-labelsNAME        READY    STATUS    RESTARTS  AGE      LABELSfirst-pod    1/1      Running  0          25m      app=bash,tir=unkonwnsecond-pod  1/1      Running  0          8m        app=bash,tir=backend

可以将标签显示为列

[root@devops-101 ~]# kubectl get pods -L app,tirNAME        READY    STATUS    RESTARTS  AGE      APP      TIRfirst-pod    1/1      Running  0          26m      bash      unkonwnsecond-pod  1/1      Running  0          9m        bash      backend

标签是Kubernetes中非常强大的一个功能,Node节点也可以增加标签,再利用Pod的标签选择器,可以将Pod分配到不同类型的Node上。

1.3.5 删除Pod

[root@devops-101 ~]# kubectl delete pods first-podpod"first-pod"deleted

也可以根据标签选择器删除。

[root@devops-101 ~]# kubectl delete pods -l tir=backendpod"second-pod"deleted

1.4 Pod的生命周期

像单独的容器应用一样,Pod并不是持久运行的。Pod创建后,Kubernetes为其分配一个UID,并且通过Controller调度到Node中运行,然后Pod一直保持运行状态直到运行正常结束或者被删除。在Node发生故障时,Controller负责将其调度到其他的Node中。Kubernetes为Pod定义了几种状态,分别如下:

Pending,Pod已创建,正在等待容器创建。经常是正在下载镜像,因为这一步骤最耗费时间。

Running,Pod已经绑定到某个Node并且正在运行。或者可能正在进行意外中断后的重启。

Succeeded,表示Pod中的容器已经正常结束并且不需要重启。

Failed,表示Pod中的容器遇到了错误而终止。

Unknown,因为网络或其他原因,无法获取Pod的状态。

2. 如何对Pod进行健康检查

Kubernetes利用Handler功能,可以对容器的状况进行探测,有以下三种形式。

ExecAction:在容器中执行特定的命令。

TCPSocketAction:检查容器端口是否可以连接。

HTTPGetAction:检查HTTP请求状态是否正常。

这部分内容展开来也比较多,这部分的内容参考Kubernetes中Pod的健康检查

3. Init Containers

Pod中可以包含一到多个Init Container,在其他容器之前开始运行。Init Container 只能是运行到完成状态,即不能够一直存在。Init Container必须依次执行。在App Container运行前,所有的Init Container必须全部正常结束。

在Pod启动过程中,Init Container在网络和存储初始化完成后开始按顺序启动。Pod重启的时候,所有的Init Container都会重新执行。

However, if the Pod restartPolicy is set to Always, the Init Containers use RestartPolicy OnFailure.

3.1 好处

运行一些不希望在 App Container 中运行的命令或工具

包含一些App Image中没有的工具或特定代码

应用镜像构建人员和部署人员可以独立工作而不需要依赖对方

拥有与App Container不同的命名空间

因为在App Container运行前必须运行结束,适合做一些前置条件的检查和配置

3.2 语法

先看一下解释

[root@devops-101 ~]# kubectl explain pod.spec.initContainersKIND:    PodVERSION:  v1RESOURCE: initContainers <[]Object>DESCRIPTION:    List of initialization containers belonging to the pod. Init containers are    executedinorder prior to containers being started. If any init container    fails, the pod is considered to have failed and is handled according to its    restartPolicy. The nameforan init container or normal container must be    unique among all containers. Init containers may not have Lifecycle    actions, Readiness probes, or Liveness probes. The resourceRequirements of    an init container are taken into account during scheduling by finding the    highest request/limitforeach resourcetype, andthenusing the max of of    that value or the sum of the normal containers. Limits are applied to init    containersina similar fashion. Init containers cannot currently be added    or removed. Cannot be updated. More info:    https://kubernetes.io/docs/concepts/workloads/pods/init-containers/    A single application container that you want to run within a pod.

具体语法。

apiVersion:v1kind:Podmetadata:  name:myapp-pod  labels:    app:myappspec:  containers:  - name:myapp-container    image:docker.io/busybox    command:['sh','-c','echo The app is running! && sleep 3600']  initContainers:  - name:init-myservice    image:docker.io/busybox    command:['sh','-c','echo init-service && sleep 2']  - name:init-mydb    image:docker.io/busybox    command:['sh','-c','echo init-mydb && sleep 2']

兼容性问题

1.5之前的语法都写在 annotation 中,1.6 以上的版本使用 .spec.initContainers 字段。建议还是使用 1.6 版本的语法。1.6、1.7的版本还兼容1.5以下的版本,1.8之后就不再兼容老版本了。

4. Pod Preset

利用这个特性,可以在Pod启动过程中向Pod中注入密码 Secrets、存储 Volumes、挂载点 Volume Mounts和环境变量。通过标签选择器来指定Pod。利用这个特性,Pod Template的维护人员就不需要为每个Pod显示的提供相关的属性。

具体的工作步骤

检查所有可用的ProdPresets

检查是否有ProdPreset的标签与即将创建的Pod相匹配

将PodPreset中定义的参数与Pod定义合并

如果参数合并出错,则丢弃ProPreset参数,继续创建Pod

为Pod增加注解,表示层被ProdPreset修改过,形式为 podpreset.admission.kubernetes.io/podpreset-<pod-preset name>: "<resource version>"

对于 Env、EnvFrom、VolumeMounts Kubernetes修改Container Spec,对于Volume修改Pod Spec。

4.1 对个别Pod停用

在Spec中增加注解:

podpreset.admission.kubernetes.io/exclude:"true"

5. 中断

Pod会因为各种各样的原因发生中断。

5.1 计划内中断

删除部署 Deployment或者其他控制器

更新部署模版导致的Pod重启

直接删除Pod

集群的缩容

手工移除

5.2 计划外中断

硬件故障、物理节点宕机

集群管理员误删VM

云供应商故障导致的主机不可用

Kernel panic

集群网络分区导致节点消失

资源耗尽导致的节点剔除

5.3 PDB Disruption Budgets

Kubernetes offers features to help run highly available applications at the same time as frequent voluntary disruptions. We call this set of features Disruption Budgets.

Kubernetes允许我们创建一个PDB对象,来确保一个RS中运行的Pod不会在一个预算(个数)之下。

Eviction API。

PDB是用来解决集群管理和应用管理职责分离的情况,如果你的单位不存在这种情况,就可以不使用PDB。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,524评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,869评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,813评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,210评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,085评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,117评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,533评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,219评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,487评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,582评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,362评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,218评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,589评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,899评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,176评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,503评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,707评论 2 335