A Big Picture of Kubernetes
作者delphis,来自腾讯音乐全民K歌基础架构团队,致力于推动Service Mesh、Kubernetes等云原生技术在部门内的落地和实践。
1. 前言
互联网后台架构技术的发展一日千里。身处技术变革浪潮的后台同学,应该都深切地感受到了云原生技术在公司内外的蓬勃发展。云原生技术正在逐渐成为后台工程师与架构师们的必修课,而 kubernetes 正是云原生的基石,聚光灯下的焦点。
kubernetes 是一个被写了很多次的主题,本文并不希望事无巨细地阐述其所有内容。事实上,仅凭一篇文章的篇幅也无法写透这个宏大的主题。即便写出来,也会变成毫无重点的堆砌,很难快速消化吸收。因此,本文更倾向于作为 kubernetes 入门的一张 Big Picture,记录笔者在接触 kubernetes 的过程中关注的那些问题点。
由于从业经历的不同,不同人在陈述同一个主题时,切入的角度往往有所不同。举例来说:不同的互联网公司(特别是头部公司),通常有自己偏爱的技术文化。譬如在进程间通信这个方向,Amazon 就比较推崇 Service Interfaces(没办法,老板喜欢)。而国内某大厂的同学在做技术选型时,则更偏爱采用 Shared Memory。
2. 何为云原生与云原生应用?
学习一个事物时,通常都需要先做基础的两点功课:首先要了解它的背景与外延,其次了解它的内涵。 首先了解它的外延,是为了分辨它在整张 Big Picture 中的位置。Kubernetes 的背景,就是云原生技术。于是,我们不禁要问几个问题:
何为云原生? 什么样的应用才能称作“云原生应用”? 云原生应用与传统后台应用有何区别?
2.1 何为云原生?
Cloud Native 最早是在 2013 年由 Pivotal 公司的 Matt Stine 提出的。2015 年 CNCF(Cloud Native Computing Foudation,云原生计算基金会)成立。官方发布的云原生 v1.0 定义是:“云原生技术有利于各组织在公有云、私有云和混合云等新型动态环境中,构建和运行可弹性扩展的应用。云原生的代表技术包括容器、服务网格、微服务、不可变基础设施和声明式 API。 这些技术能够构建容错性好、易于管理和便于观察的松耦合系统。结合可靠的自动化手段,云原生技术使工程师能够轻松地对系统作出频繁和可预测的重大变更。” 在该定义中,容器、不可变基础设施、声明式 API 都与 kubernetes 直接相关。
2.2 何为云原生应用?
按字面意思的理解,云原生应用是指在云上生长出来的应用,云上的“原住民”。然而这也没有解释它与传统应用的区别,也没有说明它为何更“高级”?2017 年,Red Hat 架构师、《Kubernetes Patterns》的作者 Bilgin Lbryam 给云原生应用下了一个比较准确而连贯的定义(参见文献[3]):
以微服务原则进行划分设计。
使用 devops 和 CI/CD 的方式进行开发和交付。
以容器技术进行打包发布。
在云基础设施上运行并被调度。
2.3 小结
云原生是当前互联网后台一个非常具有前景的技术领域。首先,这片土地足够广阔,可以让每一个后台同学去学习与深耕。其次,这个方向也足够主流与实用,看看业内如火如荼的各种技术峰会、培训课、岗位招聘。云原生不是那种没有使用价值的“屠龙之技”,值得深入去钻研。
3. 何为 k8s?提供什么能力?解决什么问题?
本节来了解 kubernetes 的内涵,即它涵盖了哪些内容,提供了哪些能力。如果说 istio 是一艘快艇的话,k8s 就是一艘巨轮,驰骋在更广阔的海域。 用最简短的语言描述 K8S,可以说,K8S 是一个容器编排系统(Container Ochestration)。那么,“编排”二字包含了哪些核心内容?
镜像管理(Image): 管理与分发。 容器管理(Container): 创建、调度、状态监控、自动伸缩。 服务管理(Service): 发布升级、服务发现与负载均衡。 配置管理(Config): ConfigMap 是普通配置,Secret 是敏感数据。配置也是非常重要的,有很多细节可以深入展开。 流量治理(Traffic): 日志、监控、鉴权等。
4. k8s 集群架构
上文是从外部视角去描述并确定我们讨论的这个主题,kubernetes 的边界。本节将描述 kubernetes 集群的内部结构。相信后台同学看完之后,都会有似曾相识的感觉。
4.1 Master 组件
kube-apiserver: 对外暴露可以操作整个 kubernetes 集群的 REST API。 kube-scheduler: 负责调度 worker 上的 pods。 kube-controller-manager: 管理各种 kubernetes 定义的 controller。 etcd: Key-Value 存储组件,采用 Raft 协议,存储集群的各种状态数据,包括配置、节点、Pod 等。
4.2 Worker/Node 组件
kubelet: 是一个 Agent,监控 node 上的 container 是否正常运行。 kube-proxy: 操纵机器上的 iptables 网络规则,执行转发。 container runtime: 容器运行的基础环境,负责下载镜像与运行容器。
5. 问答系列:Questions about k8s?
真理越辩越明。很多看似理所当然的东西,背后都是作者们经过深思熟虑后的选择。 想对某个技术领域有深入了解,需要日夜泡在其中。一个比较实用的方式是不停地对自己发问,姑且称之为“问题反馈闭环”。按照国外大佬们的造词功力,可能叫做“沉思录”。前面几个小节只是一枚敲门砖,可以说是让我们站在门口。只有不断发问与复盘,才能慢慢迈过门槛,走进深水区。
5.1 有了 kubernetes,为何还需要 istio?
答:kubernetes 并不提供精细化的流量调度能力,例如精细化路由、分布式限流等。
5.2 GKE (Google Kubernetes Engine) 与 K8S 的区别?
答:GKE 只是托管 K8S 集群的一个平台,面向企业与用户提供快速搭建与维护自己 K8S 集群的能力。业界还有阿里的 ACK,腾讯的 TKE,华为的 CCE 等竞品。有个说法很形象:K8S 只是一套毛坯样板,而像 GKE 这样的平台则相当于房地产商,开发并出售一套套精装修的商品房,让你可以拎包入住。
GKE 是开箱即用(Out-of-Box)的: 做好了控制台页面,客户只需要点击就能完成自己的 k8s 集群的创建。 GKE 是多租户的: 面向不同的企业和用户。
5.3 何为不可变基础设施(Immutable Infrastructure)?
答:不得不佩服西方人的抽象能力。具体定义参见文献[4]。
5.4 何为声明式 API(Declarative API)?
答:同样是云原生的八大原则之一。提起声明式,是不是想起了 SQL 这款声明式查询语言?参见文献[8]。
5.5 一个 K8S 集群的最大规模(最多可以容纳多少 Node、Pod、Container)?
答:k8s 官方有一个页面专门回答了这个问题,参见文献[10]。可以进一步追问这个问题,制约集群规模的瓶颈是哪个部分?CPU/存储/数据同步?
5.6 为何推荐一个容器只部署一个进程?
答:最主要应该还是设计思想的考虑,就是倡导一个容器只做一件事。其次是为了解耦,因为在同一个容器内,一个进程的挂掉会导致容器杀掉其他所有进程。
5.7 同一个 Pod 内的容器可否使用共享内存通信?
答:可以。同一个 Pod 内的容器共享同一个 IPC 命名空间,它们可以使用标准的进程间通信方式来互相通信,比如“SystemV 信号量”与“POSIX 共享内存”。
5.8 同一个 Pod 内的容器可否使用 Unix Domain Socket 通信?
答:可以。同一个 Pod 内的容器是共享网络与存储的。因此,不仅可以使用 UDS 通信,也可以支持部署一个日志 Agent 采集同一个 Pod 内的业务服务的日志。
5.9 K8S 能否根据机器负载进行自动扩缩容,而不是人工调整 replica 数量?
答:可以。该特性称作 HPA (Horizontal Pod Autoscaling),还有一个与之对称的概念 VPA(Vertical Pod Autoscaling)。
5.10 K8S 为何选择 etcd 作为数据存储,而不是其他分布式 KV 存储?
答:k8s 使用 etcd 存储集群的 API objects、服务发现、配置与状态数据。etcd 拥有如下特点,可以说是一个比较全面的选手:
持久化能力: 有些 KV 缓存并不具备该能力,比如 memcache。 数据一致性 高可用 高性能 安全性: 支持基于 TLS 与 SSL 的鉴权。也可以看看 etcd 官网自己是怎么说的。最后可能还有一点,etcd 是使用 golang 开发的,是 Clouad Native 阵营里的“自己人”。
6. 小结
本文提供了一个切入 kubernetes 的角度,在一定程度上破除“kubernetes 很复杂”的印象。为了避免篇幅过长,本文重点回答了 What 与 Why 的问题(什么是 kubernetes?为什么需要它?拿它来做什么?),而没有回答 How 的问题。距离 kubernetes v1.0 发布已经过去了 6 年,这片水面下的冰山不可谓不深,需要上下求索。万里长征第一步,相信我们已经有了一个不错的开始。
7. 参考文献
Last But Not Least. 参考文献虽然放在最后,却不是无关紧要的;相反它们可能是最有价值的部分。
[1] CNCF Cloud Native Definition.
https://github.com/cncf/toc/blob/main/DEFINITION.md
[2] 云原生的发展历程.
https://erdong.site/cloudnative-notes/chapterA-01-Basic/A-1.1-cloud-native-history.html
[3] designing cloud native applications.
https://www.slideshare.net/bibryam/designing-cloud-native-applications-with-kubernetes
[4] 不可变基础设施.
https://blog.csdn.net/chaocai2004/article/details/103827372
[5] What is kubernetes.
https://www.bmc.com/blogs/what-is-kubernetes/
[6] 读张磊的《深度剖析 Kubernetes》.
https://skyscribe.github.io/post/2019/08/24/learn-kubernetes-again/
[7] SpringBoot 与 Kubernetes 云原生微服务实战.
https://www.bilibili.com/video/BV1gf4y1s769?p=73
[8] Kubernetes 核心理解之声明式 API 和和编程范式.
https://blog.csdn.net/KevinBetterQ/article/details/104012293
[9] difference between kubernetes and gke.
https://stackoverflow.com/questions/37904711/what-is-the-difference-between-kubernetes-and-gke
[10] Considerations for large clusters.
https://kubernetes.io/docs/setup/best-practices/cluster-large/
[11] etcd & kubernetes: What you should know.
https://rafay.co/the-kubernetes-current/etcd-kubernetes-what-you-should-know/
[12] Horizontal Pod Autoscaler.
https://kubernetes.io/docs/tasks/run-application/horizontal-pod-autoscale/
推荐
原创不易,随手关注或者”在看“,诚挚感谢!