gitbook/深入剖析Kubernetes/docs/40583.md
2022-09-03 22:05:03 +08:00

8.4 KiB
Raw Permalink Blame History

16 | 编排其实很简单:谈谈“控制器”模型

你好,我是张磊。今天我和你分享的主题是:编排其实很简单之谈谈“控制器”模型。

在上一篇文章中我和你详细介绍了Pod的用法讲解了Pod这个API对象的各个字段。而接下来我们就一起来看看“编排”这个Kubernetes项目最核心的功能吧。

实际上,你可能已经有所感悟:Pod这个看似复杂的API对象实际上就是对容器的进一步抽象和封装而已。

说得更形象些,“容器”镜像虽然好用,但是容器这样一个“沙盒”的概念,对于描述应用来说,还是太过简单了。这就好比,集装箱固然好用,但是如果它四面都光秃秃的,吊车还怎么把这个集装箱吊起来并摆放好呢?

所以Pod对象其实就是容器的升级版。它对容器进行了组合添加了更多的属性和字段。这就好比给集装箱四面安装了吊环使得Kubernetes这架“吊车”可以更轻松地操作它。

而Kubernetes操作这些“集装箱”的逻辑都由控制器Controller完成。在前面的第12篇文章《牛刀小试:我的第一个容器化应用》我们曾经使用过Deployment这个最基本的控制器对象。

现在我们一起来回顾一下这个名叫nginx-deployment的例子

apiVersion: apps/v1
kind: Deployment
metadata:
  name: nginx-deployment
spec:
  selector:
    matchLabels:
      app: nginx
  replicas: 2
  template:
    metadata:
      labels:
        app: nginx
    spec:
      containers:
      - name: nginx
        image: nginx:1.7.9
        ports:
        - containerPort: 80

这个Deployment定义的编排动作非常简单确保携带了app=nginx标签的Pod的个数永远等于spec.replicas指定的个数即2个。

这就意味着如果在这个集群中携带app=nginx标签的Pod的个数大于2的时候就会有旧的Pod被删除反之就会有新的Pod被创建。

这时你也许就会好奇究竟是Kubernetes项目中的哪个组件在执行这些操作呢

我在前面介绍Kubernetes架构的时候曾经提到过一个叫作kube-controller-manager的组件。

实际上这个组件就是一系列控制器的集合。我们可以查看一下Kubernetes项目的pkg/controller目录

$ cd kubernetes/pkg/controller/
$ ls -d */              
deployment/             job/                    podautoscaler/          
cloud/                  disruption/             namespace/              
replicaset/             serviceaccount/         volume/
cronjob/                garbagecollector/       nodelifecycle/          replication/            statefulset/            daemon/
...

这个目录下面的每一个控制器都以独有的方式负责某种编排功能。而我们的Deployment正是这些控制器中的一种。

实际上这些控制器之所以被统一放在pkg/controller目录下就是因为它们都遵循Kubernetes项目中的一个通用编排模式控制循环control loop

比如现在有一种待编排的对象X它有一个对应的控制器。那么我就可以用一段Go语言风格的伪代码为你描述这个控制循环

for {
  实际状态 := 获取集群中对象X的实际状态Actual State
  期望状态 := 获取集群中对象X的期望状态Desired State
  if 实际状态 == 期望状态{
    什么都不做
  } else {
    执行编排动作,将实际状态调整为期望状态
  }
}

在具体实现中实际状态往往来自于Kubernetes集群本身

比如kubelet通过心跳汇报的容器状态和节点状态或者监控系统中保存的应用监控数据或者控制器主动收集的它自己感兴趣的信息这些都是常见的实际状态的来源。

而期望状态一般来自于用户提交的YAML文件

比如Deployment对象中Replicas字段的值。很明显这些信息往往都保存在Etcd中。

接下来以Deployment为例我和你简单描述一下它对控制器模型的实现

  1. Deployment控制器从Etcd中获取到所有携带了“app: nginx”标签的Pod然后统计它们的数量这就是实际状态

  2. Deployment对象的Replicas字段的值就是期望状态

  3. Deployment控制器将两个状态做比较然后根据比较结果确定是创建Pod还是删除已有的Pod具体如何操作Pod对象我会在下一篇文章详细介绍

可以看到一个Kubernetes对象的主要编排逻辑实际上是在第三步的“对比”阶段完成的。

这个操作通常被叫作调谐Reconcile。这个调谐的过程则被称作“Reconcile Loop”调谐循环或者“Sync Loop”同步循环

所以,如果你以后在文档或者社区中碰到这些词,都不要担心,它们其实指的都是同一个东西:控制循环。

而调谐的最终结果,往往都是对被控制对象的某种写操作。

比如增加Pod删除已有的Pod或者更新Pod的某个字段。这也是Kubernetes项目“面向API对象编程”的一个直观体现。

其实像Deployment这种控制器的设计原理就是我们前面提到过的“用一种对象管理另一种对象”的“艺术”。

其中这个控制器对象本身负责定义被管理对象的期望状态。比如Deployment里的replicas=2这个字段。

而被控制对象的定义则来自于一个“模板”。比如Deployment里的template字段。

可以看到Deployment这个template字段里的内容跟一个标准的Pod对象的API定义丝毫不差。而所有被这个Deployment管理的Pod实例其实都是根据这个template字段的内容创建出来的。

像Deployment定义的template字段在Kubernetes项目中有一个专有的名字叫作PodTemplatePod模板

这个概念非常重要因为后面我要讲解到的大多数控制器都会使用PodTemplate来统一定义它所要管理的Pod。更有意思的是我们还会看到其他类型的对象模板比如Volume的模板。

至此我们就可以对Deployment以及其他类似的控制器做一个简单总结了

如上图所示,类似Deployment这样的一个控制器实际上都是由上半部分的控制器定义包括期望状态加上下半部分的被控制对象的模板组成的。

这就是为什么在所有API对象的Metadata里都有一个字段叫作ownerReference用于保存当前这个API对象的拥有者Owner的信息。

那么对于我们这个nginx-deployment来说它创建出来的Pod的ownerReference就是nginx-deployment吗或者说nginx-deployment所直接控制的就是Pod对象么

这个问题的答案,我就留到下一篇文章时再做详细解释吧。

总结

在今天这篇文章中我以Deployment为例和你详细分享了Kubernetes项目如何通过一个称作“控制器模式”controller pattern的设计方法来统一地实现对各种不同的对象或者资源进行的编排操作。

在后面的讲解中我还会讲到很多不同类型的容器编排功能比如StatefulSet、DaemonSet等等它们无一例外地都有这样一个甚至多个控制器的存在并遵循控制循环control loop的流程完成各自的编排逻辑。

实际上跟Deployment相似这些控制循环最后的执行结果要么就是创建、更新一些Pod或者其他的API对象、资源要么就是删除一些已经存在的Pod或者其他的API对象、资源

但也正是在这个统一的编排框架下,不同的控制器可以在具体执行过程中,设计不同的业务逻辑,从而达到不同的编排效果。

这个实现思路正是Kubernetes项目进行容器编排的核心原理。在此后讲解Kubernetes编排功能的文章中我都会遵循这个逻辑展开并且带你逐步领悟控制器模式在不同的容器化作业中的实现方式。

思考题

你能否说出Kubernetes使用的这个“控制器模式”跟我们平常所说的“事件驱动”有什么区别和联系吗

感谢你的收听,欢迎你给我留言,也欢迎分享给更多的朋友一起阅读。