在 Kubernetes 集群中,kube-apiserver
是一个至关重要的组件,它不仅要对外响应客户端的 HTTPS 请求,还要对内与 controller-manager
、scheduler
、kubelet
…… 等等其它组件交互。
为了保障 kube-apiserver
的稳定性,其提供了以下配置用于限流:
--max-requests-inflight
:查询类请求的最大并发数,默认 400。--max-mutating-requests-inflight
:变更类请求的最大并发数,默认 200。
虽然上述两项配置保障了总体的并发限制,但缺乏对请求的优先级划分,在某些情况下,集群管理员可能失去对 kube-apiserver
的掌控,比如某些失控的应用频繁请求 kube-apiserver
导致其过载而无法正常响应其它请求(2024 年 12 月 11 号 OpenAI 的 kubernetes 集群故障就是这种情况)。
而 APF
(API Priority and Fairness)机制则可以避免这种问题。
APF
APF
(API 优先级和公平调度)机制引入了以下两个新的对象:
FlowSchema
:用于对请求进行分类,关联一个PriorityLevelConfiguration
优先级。PriorityLevelConfiguration
:定义具体的优先级、该级别的并发数、队列行为。
请求通过 APF
的处理流程如下图所示:
请求首先匹配一个 FlowSchema
,然后被分流到关联的 PriorityLevelConfiguration
,进入该优先级配置的队列中,最后被从队列中取出处理。
FlowSchema
和 PriorityLevelConfiguration
的配置示例如下图所示:
FlowSchema
的 .spec
配置有四个部分:
distinguisherMethod
:可以设置 type 为 ByUser 或者 ByNamespace 将请求进一步分流,也可以不设置此配置。matchingPrecedence
:设置匹配的顺序,每个请求都先从matchingPrecedence
数值最低的FlowSchema
开始匹配,建议确保每个FlowSchema
的matchingPrecedence
不相同。priorityLevelConfiguration
:关联的优先级,每个FlowSchema
只能关联一个优先级设置。rules
:匹配的规则,如果某个请求匹配其中一条规则,则该请求会被这个FlowSchema
分流到关联的优先级。
PriorityLevelConfiguration
的 .spec
配置有三个部分:
type
:可以是Exempt
或者Limited
,Exempt
表示豁免请求,即该请求不会进入队列排队,而是被立即处理。exempt
:当type
是Exempt
时可以配置豁免的更多细节。-
limited
:优先级队列相关配置,包括以下子属性:borrowingLimitPercent
:此优先级可以借入的额度。lendablePercent
:此优先级可以借出的额度。nominalConcurrencyShares
:名义并发额度。借入和借出额度设置提供了更多的弹性。limitResponse
:当type
是Reject
时,多余请求被立即响应 429;当type
是Queue
时可以进一步设置队列的细节。
APF
需要设置 kube-apiserver
的 --enable-priority-and-fairness
为 true 开启(新版本默认开启),此时 --max-requests-inflight
和 --max-mutating-requests-inflight
相加的值决定了 kube-apiserver
的总并发数,总并发数和优先级的配额决定了每个优先级的并发能力。开启 APF
后,不再区分查询类请求和变更类请求,只根据 FlowSchema
进行分流。
内置的 FlowSchema 和 PriorityLevelConfiguration
为了降低用户的操作成本,kubernetes 内置了多种 FlowSchema
和 PriorityLevelConfiguration
,这些对象被划分为强制和建议两类。对于强制的配置对象,k8s 会确保该对象存在,且不受用户控制;对于建议的配置对象,同样会确保该对象存在,但是用户可以控制。
具体内置了哪些 FlowSchema
和 PriorityLevelConfiguration
如下图所示:
从这些 FlowSchema
和 PriorityLevelConfiguration
的名字就能看出来它们分流和优先级的意图,集群内部组件的请求,如:节点的监控状态更新、kubelet 其它请求、内置控制器选主、内置控制器的其它请求 … 等等都已经被囊括其中,另外还有 catch-all
和 global-default
进行兜底,确保所有请求都被分流到具体的优先级队列。
当然用户也可以定义自己的 FlowSchema
和 PriorityLevelConfiguration
。
总结
APF
在 v1.29 版本成为稳定的特性,为 kube-apiserver
的流量控制提供了更细粒度的处理机制。
(我是凌虚,关注我,无广告,专注技术,欢迎交流或为我推荐工作)
参考资料: