为 Pod 和容器管理资源

当你定义 Pod 时可以选择性地为每个容器设定所需要的资源数量。 最常见的可设定资源是 CPU 和内存(RAM)大小;此外还有其他类型的资源。

当你为 Pod 中的 Container 指定了资源 request(请求) 时, kube-scheduler 就利用该信息决定将 Pod 调度到哪个节点上。 当你为 Container 指定了资源 limit(限制) 时,kubelet 就可以确保运行的容器不会使用超出所设限制的资源。 kubelet 还会为容器预留所 request(请求) 数量的系统资源,供其使用。

请求和限制

如果 Pod 运行所在的节点具有足够的可用资源,容器可能(且可以)使用超出对应资源 request 属性所设置的资源量。

例如,如果你将容器的 memory 的请求量设置为 256 MiB,而该容器所处的 Pod 被调度到一个具有 8 GiB 内存的节点上,并且该节点上没有其他 Pod 运行,那么该容器就可以尝试使用更多的内存。

限制是另一个话题。cpu 限制和 memory 限制都由 kubelet (以及 容器运行时)来应用, 最终由内核强制执行。在 Linux 节点上,Linux 内核通过 CGroup 来强制执行限制。 cpu 限制和 memory 限制的执行行为略有不同。

cpu 限制通过 CPU 节流机制来强制执行。 当某容器接近其 cpu 限制值时,内核会基于容器的限制值来限制其对 CPU 的访问。 因此,cpu 限制是内核强制执行的一个硬性限制。容器不得使用超出其 cpu 限制所指定的 CPU 核数。

memory 限制由内核使用 OOM(内存溢出)终止机制来强制执行。 当某容器使用的内存超过其 memory 限制时,内核可以终止此容器。 然而,终止只会在内核检测到内存压力时才会发生。 因此,超配内存的容器可能不会被立即终止。 这意味着 memory 限制是被动执行的。 某容器可以使用超过其 memory 限制的内存,但如果这样做,此容器可能会被终止。

说明:

你可以使用一个 Alpha 特性 MemoryQoS 来尝试为内存添加执行更多的抢占限制 (这与 OOM Killer 的被动执行相反)。然而,由于可能会因内存饥饿造成活锁情形, 所以这种尝试操作会被卡滞

说明:

如果你为某个资源指定了限制,但不指定请求, 并且没有应用准入时机制为该资源设置默认请求, 然后 Kubernetes 复制你所指定的限制值,将其用作资源的请求值。

资源类型

CPU内存都是资源类型。每种资源类型具有其基本单位。 CPU 表达的是计算处理能力,其单位是 Kubernetes CPU。 内存的单位是字节。 对于 Linux 负载,则可以指定巨页(Huge Page)资源。 巨页是 Linux 特有的功能,节点内核在其中分配的内存块比默认页大小大得多。

例如,在默认页面大小为 4KiB 的系统上,你可以指定限制 hugepages-2Mi: 80Mi。 如果容器尝试分配 40 个 2MiB 大小的巨页(总共 80 MiB ),则分配请求会失败。

说明:

你不能过量使用 hugepages- * 资源。 这与 memorycpu 资源不同。

CPU 和内存统称为计算资源,或简称为资源。 计算资源的数量是可测量的,可以被请求、被分配、被消耗。 它们与 API 资源不同。 API 资源(如 Pod 和 Service)是可通过 Kubernetes API 服务器读取和修改的对象。

Pod 和容器的资源请求和限制

针对每个容器,你都可以指定其资源限制和请求,包括如下选项:

  • spec.containers[].resources.limits.cpu
  • spec.containers[].resources.limits.memory
  • spec.containers[].resources.limits.hugepages-<size>
  • spec.containers[].resources.requests.cpu
  • spec.containers[].resources.requests.memory
  • spec.containers[].resources.requests.hugepages-<size>

尽管你只能逐个容器地指定请求和限制值,但考虑 Pod 的总体资源请求和限制也是有用的。 对特定资源而言,Pod 的资源请求/限制是 Pod 中各容器对该类型资源的请求/限制的总和。

Pod 级资源规约

特性状态: Kubernetes v1.32 [alpha] (enabled by default: false)

从 Kubernetes 1.32 开始,你还可以在 Pod 级别指定资源请求和限制。 在 Pod 级别,Kubernetes 1.32 仅支持为特定资源类型设置资源请求或限制: cpu 和/或 memory。此特性目前处于 Alpha 阶段。在启用该特性后,Kubernetes 允许你声明 Pod 的总体资源预算,这在处理大量容器时特别有用, 因为在这种情况下准确评估单个容器的资源需求可能会很困难。 此外,它还允许 Pod 内的容器之间共享空闲资源,从而提高资源利用率。

对于一个 Pod,你可以通过包含以下内容来指定 CPU 和内存的资源限制和请求:

  • spec.resources.limits.cpu
  • spec.resources.limits.memory
  • spec.resources.requests.cpu
  • spec.resources.requests.memory

Kubernetes 中的资源单位

CPU 资源单位

CPU 资源的限制和请求以 cpu 为单位。 在 Kubernetes 中,一个 CPU 等于 1 个物理 CPU 核或者 1 个虚拟核, 取决于节点是一台物理主机还是运行在某物理主机上的虚拟机。

你也可以表达带小数 CPU 的请求。 当你定义一个容器,将其 spec.containers[].resources.requests.cpu 设置为 0.5 时, 你所请求的 CPU 是你请求 1.0 CPU 时的一半。对于 CPU 资源单位, 数量表达式 0.1 等价于表达式 100m,可以看作 “100 millicpu”。 有些人说成是“一百毫核”,其实说的是同样的事情。

CPU 资源总是设置为资源的绝对数量而非相对数量值。 例如,无论容器运行在单核、双核或者 48 核的机器上,500m CPU 表示的是大约相同的算力。

说明:

Kubernetes 不允许设置精度小于 1m0.001 的 CPU 资源。 为了避免意外使用无效的 CPU 数量,当使用少于 1 个 CPU 单元时,使用 milliCPU 形式而不是十进制形式指定 CPU 单元非常有用。

例如,你有一个使用 5m0.005 核 CPU 的 Pod,并且希望减少其 CPU 资源。 通过使用十进制形式,更难发现 0.0005 CPU 是无效值,而通过使用 milliCPU 形式, 更容易发现 0.5m 是无效值。

内存资源单位

memory 的限制和请求以字节为单位。你可以使用普通的整数, 或者带有以下数量后缀的定点数字来表示内存: E、P、T、G、M、k。你也可以使用对应的 2 的幂数:Ei、Pi、Ti、Gi、Mi、Ki。 例如,以下表达式所代表的是大致相同的值:

  1. 128974848129e6129M128974848000m123Mi

请注意后缀的大小写。如果你请求 400m 临时存储,实际上所请求的是 0.4 字节。 如果有人这样设定资源请求或限制,可能他的实际想法是申请 400Mi 字节(400Mi) 或者 400M 字节。

容器资源示例

以下 Pod 有两个容器。每个容器的请求为 0.25 CPU 和 64MiB(226 字节)内存, 每个容器的资源限制为 0.5 CPU 和 128MiB 内存。 你可以认为该 Pod 的资源请求为 0.5 CPU 和 128 MiB 内存,资源限制为 1 CPU 和 256MiB 内存。

  1. ---
  2. apiVersion: v1
  3. kind: Pod
  4. metadata:
  5. name: frontend
  6. spec:
  7. containers:
  8. - name: app
  9. image: images.my-company.example/app:v4
  10. resources:
  11. requests:
  12. memory: "64Mi"
  13. cpu: "250m"
  14. limits:
  15. memory: "128Mi"
  16. cpu: "500m"
  17. - name: log-aggregator
  18. image: images.my-company.example/log-aggregator:v6
  19. resources:
  20. requests:
  21. memory: "64Mi"
  22. cpu: "250m"
  23. limits:
  24. memory: "128Mi"
  25. cpu: "500m"

Pod 资源示例

特性状态: Kubernetes v1.32 [alpha] (enabled by default: false)

以下 Pod 明确请求了 1 个 CPU 和 100 MiB 的内存,并设置了明确的限制为 1 个 CPU 和 200 MiB 的内存。 pod-resources-demo-ctr-1 容器设置了明确的资源请求和限制。然而,pod-resources-demo-ctr-2 容器没有设置明确的资源请求和限制,因此它将共享 Pod 资源边界内的可用资源。

  1. pods/resource/pod-level-resources.yaml
  1. apiVersion: v1
  2. kind: Pod
  3. metadata:
  4. name: pod-resources-demo
  5. namespace: pod-resources-example
  6. spec:
  7. resources:
  8. limits:
  9. cpu: "1"
  10. memory: "200Mi"
  11. requests:
  12. cpu: "1"
  13. memory: "100Mi"
  14. containers:
  15. - name: pod-resources-demo-ctr-1
  16. image: nginx
  17. resources:
  18. limits:
  19. cpu: "0.5"
  20. memory: "100Mi"
  21. requests:
  22. cpu: "0.5"
  23. memory: "50Mi"
  24. - name: pod-resources-demo-ctr-2
  25. image: fedora
  26. command:
  27. - sleep
  28. - inf

带资源请求的 Pod 如何调度

当你创建一个 Pod 时,Kubernetes 调度程序将为 Pod 选择一个节点。 每个节点对每种资源类型都有一个容量上限:可为 Pod 提供的 CPU 和内存量。 调度程序确保对于每种资源类型,所调度的容器的资源请求的总和小于节点的容量。 请注意,尽管节点上的实际内存或 CPU 资源使用量非常低,如果容量检查失败, 调度程序仍会拒绝在该节点上放置 Pod。 当稍后节点上资源用量增加,例如到达请求率的每日峰值区间时,节点上也不会出现资源不足的问题。

Kubernetes 应用资源请求与限制的方式

当 kubelet 将容器作为 Pod 的一部分启动时,它会将容器的 CPU 和内存请求与限制信息传递给容器运行时。

在 Linux 系统上,容器运行时通常会配置内核 CGroup,负责应用并实施所定义的请求。

  • CPU 限制定义的是容器可使用的 CPU 时间的硬性上限。 在每个调度周期(时间片)期间,Linux 内核检查是否已经超出该限制; 内核会在允许该 CGroup 恢复执行之前会等待。

  • CPU 请求值定义的是一个权重值。如果若干不同的容器(CGroup)需要在一个共享的系统上竞争运行, CPU 请求值大的负载会获得比请求值小的负载更多的 CPU 时间。

  • 内存请求值主要用于(Kubernetes)Pod 调度期间。在一个启用了 CGroup v2 的节点上, 容器运行时可能会使用内存请求值作为设置 memory.minmemory.low 的提示值。

  • 内存限制定义的是 CGroup 的内存限制。如果容器尝试分配的内存量超出限制, 则 Linux 内核的内存不足处理子系统会被激活,并停止尝试分配内存的容器中的某个进程。 如果该进程在容器中 PID 为 1,而容器被标记为可重新启动,则 Kubernetes 会重新启动该容器。

  • Pod 或容器的内存限制也适用于通过内存作为介质的卷,例如 emptyDir 卷。 kubelet 会跟踪 tmpfs 形式的 emptyDir 卷用量,将其作为容器的内存用量, 而不是临时存储用量。当使用内存作为介质的 emptyDir 时, 请务必查看下面的注意事项。

如果某容器内存用量超过其内存请求值并且所在节点内存不足时,容器所处的 Pod 可能被逐出

每个容器可能被允许也可能不被允许使用超过其 CPU 限制的处理时间。 但是,容器运行时不会由于 CPU 使用率过高而杀死 Pod 或容器。

要确定某容器是否会由于资源限制而无法调度或被杀死,请参阅疑难解答节。

监控计算和内存资源用量

kubelet 会将 Pod 的资源使用情况作为 Pod status 的一部分来报告的。

如果为集群配置了可选的监控工具, 则可以直接从指标 API 或者监控工具获得 Pod 的资源使用情况。

使用内存作为介质的 emptyDir 卷的注意事项

注意:

如果你没有为 emptyDir 卷指定 sizeLimit,该卷就会消耗 Pod 的内存, 卷的用量上限为 Pod 的内存限制(Pod.spec.containers[].resources.limits.memory)。 如果你没有设置内存限制,Pod 的内存消耗将没有上限,并且可能会用掉节点上的所有可用内存。 Kubernetes 基于资源请求(Pod.spec.containers[].resources.requests)调度 Pod, 并且在决定另一个 Pod 是否适合调度到某个给定的节点上时,不会考虑超出请求的内存用量。 这可能导致拒绝服务,并使操作系统出现需要处理内存不足(OOM)的情况。 用户可以创建任意数量的 emptyDir,可能会消耗节点上的所有可用内存,使得 OOM 更有可能发生。

从内存管理的角度来看,进程使用内存作为工作区与使用内存作为 emptyDir 的介质有一些相似之处。 但当将内存用作存储卷(例如内存为介质的 emptyDir 卷)时,你需要额外注意以下几点:

  • 存储在内存为介质的卷上的文件几乎完全由用户应用所管理。 与用作进程工作区的用法不同,你无法依赖语言级别垃圾回收这类机制。
  • 将文件写入某个卷的目的是保存数据或在应用之间传递数据。 Kubernetes 或操作系统都不会自动从卷中删除文件, 因此当系统或 Pod 面临内存压力时,将无法回收这些文件所使用的内存。
  • 以内存为介质的 emptyDir 因性能较好而很有用,但内存通常比其他存储介质(如磁盘或 SSD)小得多且成本更高。 为 emptyDir 卷使用大量内存可能会影响 Pod 或整个节点的正常运行,因此你应谨慎使用。

如果你在管理集群或命名空间,还可以设置限制内存使用的 ResourceQuota; 你可能还希望定义一个 LimitRange 以施加额外的限制。如果为每个 Pod 指定 spec.containers[].resources.limits.memory, 那么 emptyDir 卷的最大尺寸将是 Pod 的内存限制。

作为一种替代方案,集群管理员可以使用诸如 ValidationAdmissionPolicy 之类的策略机制来强制对新 Pod 的 emptyDir 卷进行大小限制。

本地临时存储

特性状态: Kubernetes v1.25 [stable]

节点通常还可以具有本地的临时性存储,由本地挂接的可写入设备或者有时也用 RAM 来提供支持。“临时(Ephemeral)”意味着对所存储的数据不提供长期可用性的保证。

Pods 通常可以使用临时性本地存储来实现缓冲区、保存日志等功能。 kubelet 可以为使用本地临时存储的 Pods 提供这种存储空间,允许后者使用 emptyDir 类型的将其挂载到容器中。

kubelet 也使用此类存储来保存节点层面的容器日志、 容器镜像文件以及运行中容器的可写入层。

注意:

如果节点失效,存储在临时性存储中的数据会丢失。 你的应用不能对本地临时性存储的性能 SLA(例如磁盘 IOPS)作任何假定。

说明:

为了使临时性存储的资源配额生效,需要完成以下两个步骤:

  • 管理员在命名空间中设置临时性存储的资源配额。
  • 用户需要在 Pod 规约中指定临时性存储资源的限制。

如果用户在 Pod 规约中未指定临时性存储资源的限制, 则临时性存储的资源配额不会生效。

Kubernetes 允许你跟踪、预留和限制 Pod 可消耗的临时性本地存储数量。

本地临时性存储的配置

Kubernetes 有两种方式支持节点上配置本地临时性存储:

采用这种配置时,你会把所有类型的临时性本地数据(包括 emptyDir 卷、可写入容器层、容器镜像、日志等)放到同一个文件系统中。 作为最有效的 kubelet 配置方式,这意味着该文件系统是专门提供给 Kubernetes (kubelet)来保存数据的。

kubelet 也会生成节点层面的容器日志, 并按临时性本地存储的方式对待之。

kubelet 会将日志写入到所配置的日志目录(默认为 /var/log)下的文件中; 还会针对其他本地存储的数据使用同一个基础目录(默认为 /var/lib/kubelet)。

通常,/var/lib/kubelet/var/log 都是在系统的根文件系统中。kubelet 的设计也考虑到这一点。

你的集群节点当然可以包含其他的、并非用于 Kubernetes 的很多文件系统。

你使用节点上的某个文件系统来保存运行 Pod 时产生的临时性数据:日志和 emptyDir 卷等。你可以使用这个文件系统来保存其他数据(例如:与 Kubernetes 无关的其他系统日志);这个文件系统还可以是根文件系统。

kubelet 也将节点层面的容器日志 写入到第一个文件系统中,并按临时性本地存储的方式对待之。

同时你使用另一个由不同逻辑存储设备支持的文件系统。在这种配置下,你会告诉 kubelet 将容器镜像层和可写层保存到这第二个文件系统上的某个目录中。

第一个文件系统中不包含任何镜像层和可写层数据。

当然,你的集群节点上还可以有很多其他与 Kubernetes 没有关联的文件系统。

kubelet 能够度量其本地存储的用量。 实现度量机制的前提是你已使用本地临时存储所支持的配置之一对节点进行配置。

如果你的节点配置不同于以上预期,kubelet 就无法对临时性本地存储实施资源限制。

说明:

kubelet 会将 tmpfs emptyDir 卷的用量当作容器内存用量,而不是本地临时性存储来统计。

说明:

kubelet 将仅跟踪临时存储的根文件系统。 挂载一个单独磁盘到 /var/lib/kubelet/var/lib/containers 的操作系统布局将不会正确地报告临时存储。

为本地临时性存储设置请求和限制

你可以指定 ephemeral-storage 来管理本地临时性存储。 Pod 中的每个容器可以设置以下属性:

  • spec.containers[].resources.limits.ephemeral-storage
  • spec.containers[].resources.requests.ephemeral-storage

ephemeral-storage 的请求和限制是按量纲计量的。 你可以使用一般整数或者定点数字加上下面的后缀来表达存储量:E、P、T、G、M、k。 你也可以使用对应的 2 的幂级数来表达:Ei、Pi、Ti、Gi、Mi、Ki。 例如,下面的表达式所表达的大致是同一个值:

  • 128974848
  • 129e6
  • 129M
  • 123Mi

请注意后缀的大小写。如果你请求 400m 临时存储,实际上所请求的是 0.4 字节。 如果有人这样设定资源请求或限制,可能他的实际想法是申请 400Mi 字节(400Mi) 或者 400M 字节。

在下面的例子中,Pod 包含两个容器。每个容器请求 2 GiB 大小的本地临时性存储。 每个容器都设置了 4 GiB 作为其本地临时性存储的限制。 因此,整个 Pod 的本地临时性存储请求是 4 GiB,且其本地临时性存储的限制为 8 GiB。 该限制值中有 500Mi 可供 emptyDir 卷使用。

  1. apiVersion: v1
  2. kind: Pod
  3. metadata:
  4. name: frontend
  5. spec:
  6. containers:
  7. - name: app
  8. image: images.my-company.example/app:v4
  9. resources:
  10. requests:
  11. ephemeral-storage: "2Gi"
  12. limits:
  13. ephemeral-storage: "4Gi"
  14. volumeMounts:
  15. - name: ephemeral
  16. mountPath: "/tmp"
  17. - name: log-aggregator
  18. image: images.my-company.example/log-aggregator:v6
  19. resources:
  20. requests:
  21. ephemeral-storage: "2Gi"
  22. limits:
  23. ephemeral-storage: "4Gi"
  24. volumeMounts:
  25. - name: ephemeral
  26. mountPath: "/tmp"
  27. volumes:
  28. - name: ephemeral
  29. emptyDir:
  30. sizeLimit: 500Mi

带临时性存储的 Pods 的调度行为

当你创建一个 Pod 时,Kubernetes 调度器会为 Pod 选择一个节点来运行之。 每个节点都有一个本地临时性存储的上限,是其可提供给 Pod 使用的总量。 欲了解更多信息, 可参考节点可分配资源节。

调度器会确保所调度的容器的资源请求总和不会超出节点的资源容量。

临时性存储消耗的管理

如果 kubelet 将本地临时性存储作为资源来管理,则 kubelet 会度量以下各处的存储用量:

  • emptyDir 卷,除了 tmpfs emptyDir
  • 保存节点层面日志的目录
  • 可写入的容器镜像层

如果某 Pod 的临时存储用量超出了你所允许的范围,kubelet 会向其发出逐出(eviction)信号,触发该 Pod 被逐出所在节点。

就容器层面的隔离而言,如果某容器的可写入镜像层和日志用量超出其存储限制, kubelet 也会将所在的 Pod 标记为逐出候选。

就 Pod 层面的隔离而言,kubelet 会将 Pod 中所有容器的限制相加,得到 Pod 存储限制的总值。如果所有容器的本地临时性存储用量总和加上 Pod 的 emptyDir 卷的用量超出 Pod 存储限制,kubelet 也会将该 Pod 标记为逐出候选。

注意:

如果 kubelet 没有度量本地临时性存储的用量,即使 Pod 的本地存储用量超出其限制也不会被逐出。

不过,如果用于可写入容器镜像层、节点层面日志或者 emptyDir 卷的文件系统中可用空间太少, 节点会为自身设置本地存储不足的污点标签。 这一污点会触发对那些无法容忍该污点的 Pod 的逐出操作。

关于临时性本地存储的配置信息,请参考这里

kubelet 支持使用不同方式来度量 Pod 的存储用量:

kubelet 按预定周期执行扫描操作,检查 emptyDir 卷、容器日志目录以及可写入容器镜像层。

这一扫描会度量存储空间用量。

说明:

项目配额(Project Quota)是一个操作系统层的功能特性,用来管理文件系统中的存储用量。 在 Kubernetes 中,你可以启用项目配额以监视存储用量。 你需要确保节点上为 emptyDir 提供存储的文件系统支持项目配额。 例如,XFS 和 ext4fs 文件系统都支持项目配额。

说明:

项目配额可以帮你监视存储用量,但无法强制执行限制。

Kubernetes 所使用的项目 ID 始于 1048576。 所使用的 IDs 会注册在 /etc/projects/etc/projid 文件中。 如果该范围中的项目 ID 已经在系统中被用于其他目的,则已占用的项目 ID 也必须注册到 /etc/projects/etc/projid 中,这样 Kubernetes 才不会使用它们。

配额方式与目录扫描方式相比速度更快,结果更精确。当某个目录被分配给某个项目时, 该目录下所创建的所有文件都属于该项目,内核只需要跟踪该项目中的文件所使用的存储块个数。 如果某文件被创建后又被删除,但对应文件描述符仍处于打开状态, 该文件会继续耗用存储空间。配额跟踪技术能够精确第记录对应存储空间的状态, 而目录扫描方式会忽略被删除文件所占用的空间。

要使用配额来跟踪 Pod 的资源使用情况,Pod 必须位于用户命名空间中。 在用户命名空间内,内核限制对文件系统上 projectID 的更改,从而确保按配额计算的存储指标的可靠性。

如果你希望使用项目配额,你需要:

  • kubelet 配置中使用 featureGates 字段启用 LocalStorageCapacityIsolationFSQuotaMonitoring=true 特性门控

  • 确保 UserNamespacesSupport 特性门控已启用, 并且内核、CRI 实现和 OCI 运行时支持用户命名空间。

  • 确保根文件系统(或者可选的运行时文件系统)启用了项目配额。所有 XFS 文件系统都支持项目配额。 对 extf 文件系统而言,你需要在文件系统尚未被挂载时启用项目配额跟踪特性:

    1. # 对 ext4 而言,在 /dev/block-device 尚未被挂载时执行下面操作
    2. sudo tune2fs -O project -Q prjquota /dev/block-device
  • 确保根文件系统(或者可选的运行时文件系统)在挂载时项目配额特性是被启用了的。 对于 XFS 和 ext4fs 而言,对应的挂载选项称作 prjquota

如果不想使用项目配额,你应该:

扩展资源(Extended Resources)

扩展资源是 kubernetes.io 域名之外的标准资源名称。 它们使得集群管理员能够颁布非 Kubernetes 内置资源,而用户可以使用他们。

使用扩展资源需要两个步骤。首先,集群管理员必须颁布扩展资源。 其次,用户必须在 Pod 中请求扩展资源。

管理扩展资源

节点级扩展资源

节点级扩展资源绑定到节点。

设备插件管理的资源

有关如何颁布在各节点上由设备插件所管理的资源, 请参阅设备插件

其他资源

为了颁布新的节点级扩展资源,集群操作员可以向 API 服务器提交 PATCH HTTP 请求, 以在集群中节点的 status.capacity 中为其配置可用数量。 完成此操作后,节点的 status.capacity 字段中将包含新资源。 kubelet 会异步地对 status.allocatable 字段执行自动更新操作,使之包含新资源。

由于调度器在评估 Pod 是否适合在某节点上执行时会使用节点的 status.allocatable 值, 调度器只会考虑异步更新之后的新值。 在更新节点容量使之包含新资源之后和请求该资源的第一个 Pod 被调度到该节点之间, 可能会有短暂的延迟。

示例:

这是一个示例,显示了如何使用 curl 构造 HTTP 请求,公告主节点为 k8s-master 的节点 k8s-node-1 上存在五个 example.com/foo 资源。

  1. curl --header "Content-Type: application/json-patch+json" \
  2. --request PATCH \
  3. --data '[{"op": "add", "path": "/status/capacity/example.com~1foo", "value": "5"}]' \
  4. http://k8s-master:8080/api/v1/nodes/k8s-node-1/status

说明:

在前面的请求中,~1 是在 patch 路径中对字符 / 的编码。 JSON-Patch 中的操作路径的值被视为 JSON-Pointer 类型。 有关更多详细信息,请参见 IETF RFC 6901 第 3 节

集群层面的扩展资源

集群层面的扩展资源并不绑定到具体节点。 它们通常由调度器扩展程序(Scheduler Extenders)管理,这些程序处理资源消耗和资源配额。

你可以在调度器配置 中指定由调度器扩展程序处理的扩展资源。

示例:

下面的调度器策略配置标明集群层扩展资源 “example.com/foo” 由调度器扩展程序处理。

  • 仅当 Pod 请求 “example.com/foo” 时,调度器才会将 Pod 发送到调度器扩展程序。
  • ignoredByScheduler 字段指定调度器不要在其 PodFitsResources 断言中检查 “example.com/foo” 资源。
  1. {
  2. "kind": "Policy",
  3. "apiVersion": "v1",
  4. "extenders": [
  5. {
  6. "urlPrefix": "<extender-endpoint>",
  7. "bindVerb": "bind",
  8. "managedResources": [
  9. {
  10. "name": "example.com/foo",
  11. "ignoredByScheduler": true
  12. }
  13. ]
  14. }
  15. ]
  16. }

使用扩展资源

就像 CPU 和内存一样,用户可以在 Pod 的规约中使用扩展资源。 调度器负责资源的核算,确保同时分配给 Pod 的资源总量不会超过可用数量。

API 服务器将扩展资源的数量限制为整数。 有效数量的示例是 33000m3Ki无效数量的示例是 0.51500m(因为 1500m 结果等同于 1.5)。

说明:

扩展资源取代了非透明整数资源(Opaque Integer Resources,OIR)。 用户可以使用 kubernetes.io(保留)以外的任何域名前缀。

要在 Pod 中使用扩展资源,请在容器规约的 spec.containers[].resources.limits 映射中包含资源名称作为键。

说明:

扩展资源不能过量使用,因此如果容器规约中同时存在请求和限制,则它们的取值必须相同。

仅当所有资源请求(包括 CPU、内存和任何扩展资源)都被满足时,Pod 才能被调度。 在资源请求无法满足时,Pod 会保持在 PENDING 状态。

示例:

下面的 Pod 请求 2 个 CPU 和 1 个 “example.com/foo”(扩展资源)。

  1. apiVersion: v1
  2. kind: Pod
  3. metadata:
  4. name: my-pod
  5. spec:
  6. containers:
  7. - name: my-container
  8. image: myimage
  9. resources:
  10. requests:
  11. cpu: 2
  12. example.com/foo: 1
  13. limits:
  14. example.com/foo: 1

PID 限制

进程 ID(PID)限制允许对 kubelet 进行配置,以限制给定 Pod 可以消耗的 PID 数量。 有关信息,请参见 PID 限制

疑难解答

我的 Pod 处于悬决状态且事件信息显示 FailedScheduling

如果调度器找不到该 Pod 可以匹配的任何节点,则该 Pod 将保持未被调度状态, 直到找到一个可以被调度到的位置。每当调度器找不到 Pod 可以调度的地方时, 会产生一个 Event。 你可以使用 kubectl 来查看 Pod 的事件;例如:

  1. kubectl describe pod frontend | grep -A 9999999999 Events
  1. Events:
  2. Type Reason Age From Message
  3. ---- ------ ---- ---- -------
  4. Warning FailedScheduling 23s default-scheduler 0/42 nodes available: insufficient cpu

在上述示例中,由于节点上的 CPU 资源不足,名为 “frontend” 的 Pod 无法被调度。 由于内存不足(PodExceedsFreeMemory)而导致失败时,也有类似的错误消息。 一般来说,如果 Pod 处于悬决状态且有这种类型的消息时,你可以尝试如下几件事情:

  • 向集群添加更多节点。
  • 终止不需要的 Pod,为悬决的 Pod 腾出空间。
  • 检查 Pod 所需的资源是否超出所有节点的资源容量。例如,如果所有节点的容量都是 cpu:1, 那么一个请求为 cpu: 1.1 的 Pod 永远不会被调度。
  • 检查节点上的污点设置。如果集群中节点上存在污点,而新的 Pod 不能容忍污点, 调度器只会考虑将 Pod 调度到不带有该污点的节点上。

你可以使用 kubectl describe nodes 命令检查节点容量和已分配的资源数量。例如:

  1. kubectl describe nodes e2e-test-node-pool-4lw4
  1. Name: e2e-test-node-pool-4lw4
  2. [ ... 这里忽略了若干行以便阅读 ...]
  3. Capacity:
  4. cpu: 2
  5. memory: 7679792Ki
  6. pods: 110
  7. Allocatable:
  8. cpu: 1800m
  9. memory: 7474992Ki
  10. pods: 110
  11. [ ... 这里忽略了若干行以便阅读 ...]
  12. Non-terminated Pods: (5 in total)
  13. Namespace Name CPU Requests CPU Limits Memory Requests Memory Limits
  14. --------- ---- ------------ ---------- --------------- -------------
  15. kube-system fluentd-gcp-v1.38-28bv1 100m (5%) 0 (0%) 200Mi (2%) 200Mi (2%)
  16. kube-system kube-dns-3297075139-61lj3 260m (13%) 0 (0%) 100Mi (1%) 170Mi (2%)
  17. kube-system kube-proxy-e2e-test-... 100m (5%) 0 (0%) 0 (0%) 0 (0%)
  18. kube-system monitoring-influxdb-grafana-v4-z1m12 200m (10%) 200m (10%) 600Mi (8%) 600Mi (8%)
  19. kube-system node-problem-detector-v0.1-fj7m3 20m (1%) 200m (10%) 20Mi (0%) 100Mi (1%)
  20. Allocated resources:
  21. (Total limits may be over 100 percent, i.e., overcommitted.)
  22. CPU Requests CPU Limits Memory Requests Memory Limits
  23. ------------ ---------- --------------- -------------
  24. 680m (34%) 400m (20%) 920Mi (11%) 1070Mi (13%)

在上面的输出中,你可以看到如果 Pod 请求超过 1.120 CPU 或者 6.23Gi 内存,节点将无法满足。

通过查看 “Pods” 部分,你将看到哪些 Pod 占用了节点上的资源。

Pods 可用的资源量低于节点的资源总量,因为系统守护进程也会使用一部分可用资源。 在 Kubernetes API 中,每个 Node 都有一个 .status.allocatable 字段 (详情参见 NodeStatus)。

字段 .status.allocatable 描述节点上可以用于 Pod 的资源总量(例如:15 个虚拟 CPU、7538 MiB 内存)。关于 Kubernetes 中节点可分配资源的信息, 可参阅为系统守护进程预留计算资源

你可以配置资源配额功能特性以限制每个名字空间可以使用的资源总量。 当某名字空间中存在 ResourceQuota 时,Kubernetes 会在该名字空间中的对象强制实施配额。 例如,如果你为不同的团队分配名字空间,你可以为这些名字空间添加 ResourceQuota。 设置资源配额有助于防止一个团队占用太多资源,以至于这种占用会影响其他团队。

你还需要考虑为这些名字空间设置授权访问: 为名字空间提供全部的写权限时,具有合适权限的人可能删除所有资源, 包括所配置的 ResourceQuota。

我的容器被终止了

你的容器可能因为资源紧张而被终止。要查看容器是否因为遇到资源限制而被杀死, 请针对相关的 Pod 执行 kubectl describe pod

  1. kubectl describe pod simmemleak-hra99

输出类似于:

  1. Name: simmemleak-hra99
  2. Namespace: default
  3. Image(s): saadali/simmemleak
  4. Node: kubernetes-node-tf0f/10.240.216.66
  5. Labels: name=simmemleak
  6. Status: Running
  7. Reason:
  8. Message:
  9. IP: 10.244.2.75
  10. Containers:
  11. simmemleak:
  12. Image: saadali/simmemleak:latest
  13. Limits:
  14. cpu: 100m
  15. memory: 50Mi
  16. State: Running
  17. Started: Tue, 07 Jul 2019 12:54:41 -0700
  18. Last State: Terminated
  19. Reason: OOMKilled
  20. Exit Code: 137
  21. Started: Fri, 07 Jul 2019 12:54:30 -0700
  22. Finished: Fri, 07 Jul 2019 12:54:33 -0700
  23. Ready: False
  24. Restart Count: 5
  25. Conditions:
  26. Type Status
  27. Ready False
  28. Events:
  29. Type Reason Age From Message
  30. ---- ------ ---- ---- -------
  31. Normal Scheduled 42s default-scheduler Successfully assigned simmemleak-hra99 to kubernetes-node-tf0f
  32. Normal Pulled 41s kubelet Container image "saadali/simmemleak:latest" already present on machine
  33. Normal Created 41s kubelet Created container simmemleak
  34. Normal Started 40s kubelet Started container simmemleak
  35. Normal Killing 32s kubelet Killing container with id ead3fb35-5cf5-44ed-9ae1-488115be66c6: Need to kill Pod

在上面的例子中,Restart Count: 5 意味着 Pod 中的 simmemleak 容器被终止并且(到目前为止)重启了五次。 原因 OOMKilled 显示容器尝试使用超出其限制的内存量。

你接下来要做的或许是检查应用代码,看看是否存在内存泄露。 如果你发现应用的行为与你所预期的相同,则可以考虑为该容器设置一个更高的内存限制 (也可能需要设置请求值)。

接下来

最后修改 December 24, 2024 at 8:45 PM PST: [zh-cn]sync manage-resources-containers quota-api-object (b280c0d40e)