website/content/zh/docs/concepts/architecture/nodes.md

30 KiB
Raw Blame History

title content_type weight
节点 concept 10

在 Kubernetes 中节点Node是执行工作的机器以前叫做 minion。根据你的集群环境,节点可以是一个虚拟机或者物理机器。每个节点都包含用于运行 pods 的必要服务,并由主控组件管理。节点上的服务包括 容器运行时、kubelet 和 kube-proxy。查阅架构设计文档中 Kubernetes 节点 一节获取更多细节。

节点状态

一个节点的状态包含以下信息:

可以使用以下命令显示节点状态和有关节点的其他详细信息:

kubectl describe node <insert-node-name-here>

下面对每个章节进行详细描述。

地址

这些字段组合的用法取决于你的云服务商或者裸机配置。

  • HostName由节点的内核设置。可以通过 kubelet 的 --hostname-override 参数覆盖。
  • ExternalIP通常是节点的可以外部路由从集群外可访问的 IP 地址。
  • InternalIP通常是节点的仅可在集群内部路由的 IP 地址。

条件

conditions 字段描述了所有 Running 节点的状态。条件的示例包括:

节点条件 描述
OutOfDisk True 表示节点的空闲空间不足以用于添加新 Pods, 否则为 False
Ready 表示节点是健康的并已经准备好接收 PodsFalse 表示节点不健康而且不能接收 PodsUnknown 表示节点控制器在最近 node-monitor-grace-period 期间(默认 40 秒)没有收到节点的消息
MemoryPressure True 表示节点存在内存压力,即节点内存可用量低,否则为 False
PIDPressure True 表示节点存在进程压力,即进程过多;否则为 False
DiskPressure True 表示节点存在磁盘压力,即磁盘可用量低,否则为 False
NetworkUnavailable True 表示节点网络配置不正确;否则为 False

节点条件使用 JSON 对象表示。例如,下面的响应描述了一个健康的节点。

"conditions": [
  {
    "type": "Ready",
    "status": "True",
    "reason": "KubeletReady",
    "message": "kubelet is posting ready status",
    "lastHeartbeatTime": "2019-06-05T18:38:35Z",
    "lastTransitionTime": "2019-06-05T11:41:27Z"
  }
]

如果 Ready 条件处于状态 Unknown 或者 False 的时间超过了 pod-eviction-timeout (一个传递给 kube-controller-manager 的参数), 节点上的所有 Pods 都会被节点控制器计划删除。默认的逐出超时时长为 5 分钟。 某些情况下当节点不可访问时apiserver 不能和其上的 kubelet 通信。 删除 Pods 的决定不能传达给 kubelet直到它重新建立和 apiserver 的连接为止。 与此同时,被计划删除的 Pods 可能会继续在游离的节点上运行。

在 1.5 版本之前的 Kubernetes 上,节点控制器会将不能访问的 Pods 从 apiserver 中 强制删除。 但在 1.5 或更高的版本里,在节点控制器确认这些 Pods 在集群中已经停止运行前,不会强制删除它们。 你可以看到这些可能在无法访问的节点上运行的 Pods 处于 Terminating 或者 Unknown 状态。 如果 kubernetes 不能基于下层基础设施推断出某节点是否已经永久离开了集群,集群管理员可能需要手动删除该节点对象。 从 Kubernetes 删除节点对象将导致 apiserver 删除节点上所有运行的 Pod 对象并释放它们的名字。

节点生命周期控制器会自动创建代表条件的污点。 当调度器将 Pod 指派给某节点时,调度器会考虑节点上的污点,但是 Pod 可以容忍的污点除外。

容量与可分配

描述节点上的可用资源CPU、内存和可以调度到节点上的 Pods 的个数上限。

capacity 块中的字段指示节点拥有的资源总量。allocatable 块指示节点上可供普通 Pod 消耗的资源量。

可以在学习如何在节点上保留计算资源 的同时阅读有关容量和可分配资源的更多信息。

信息

关于节点的通用信息例如内核版本、Kubernetes 版本kubelet 和 kube-proxy 版本、Docker 版本 (如果使用了)和操作系统名称。这些信息由 kubelet 从节点上搜集而来。

管理

PodsServices 不同, 节点并不是在 Kubernetes 中从头创建的:它们由外部的云服务商(例如 Google Compute Engine创建或者是来自你的资源池 中的物理机或者虚拟机。 这意味着当 Kubernetes 创建一个节点时,它其实仅仅创建了一个对象来代表这个节点。 创建以后Kubernetes 将检查这个节点是否可用。例如,如果你尝试使用如下内容创建一个节点:

{
  "kind": "Node",
  "apiVersion": "v1",
  "metadata": {
    "name": "10.240.79.157",
    "labels": {
      "name": "my-first-k8s-node"
    }
  }
}

Kubernetes 会在内部创一个 Node 对象(用以表示节点),并基于 metadata.name 字段执行健康检查, 如果节点合法,即所有必要服务都处于运行状态,它就有资格运行 Pod否则它将被所有的集群活动忽略直到其变为合法为止。

{{< note >}} Kubernetes 保留无效节点所对应的对象,并持续检查它是否合法。 若要停止这种检查,必须显式删除 Node 对象。 {{< /note >}}

当前,有 3 个组件同 Kubernetes 节点接口交互节点控制器、kubelet 和 kubectl。

节点控制器

节点控制器是一个 Kubernetes 控制面组件,管理节点的方方面面。

节点控制器在节点的生命周期中扮演多个角色。第一个是当节点注册时为它分配一个 CIDR 区间(如果启用了 CIDR 分配)。

第二个保持节点控制器内部的节点列表与云服务商所提供的可用机器列表同步。 如果在云环境下运行,当某节点不健康时,节点控制器将询问云服务是否节点的虚拟机可用。 如果不可用,节点控制器会将该节点从它的节点列表删除。

第三个是监控节点的健康情况。节点控制器负责在节点不可达 (即,节点控制器因为某些原因没有收到心跳,例如节点宕机)时, 将它的 NodeStatus 的 NodeReady 条件更新为 ConditionUnknown。 后续如果节点持续不可达,节点控制器将逐出节点上的所有 Pods使用体面终止。 (默认情况下 40s 后开始报告 ConditionUnknown在那之后 5m 开始逐出 Pods。 节点控制器每隔 --node-monitor-period 秒检查每个节点的状态。

心跳机制

Kubernetes 节点发送的心跳有助于确定节点的可用性。 心跳有两种形式:NodeStatus 和 [Lease 对象](/docs/reference/generated/kubernetes-api/{{< latest-version >}}/#lease-v1-coordination-k8s-io)。 每个节点在 kube-node-lease{{< glossary_tooltip term_id="namespace" text="命名空间">}} 中都有一个关联的 Lease 对象。 Lease 是一种轻量级的资源,可在集群扩展时提高节点心跳机制的性能。

kubelet 负责创建和更新 NodeStatusLease 对象。

  • 当状态发生变化时或者在配置的时间间隔内没有更新时kubelet 会更新 NodeStatusNodeStatus 更新的默认间隔为 5 分钟(比不可达节点的 40 秒默认超时时间长很多)。
  • kubelet 会每 10 秒(默认更新间隔时间)创建并更新其 Lease 对象。Lease 更新独立于 NodeStatus 更新而发生。

可靠性

在 Kubernetes 1.4 中我们更新了节点控制器逻辑以更好地处理大批量节点访问控制面而出问题的情况 (例如,控制面节点的网络出了问题)。从 1.4 开始,节点控制器在决定逐出 Pod 之前 会检查集群中所有节点的状态。

大部分情况下,节点控制器把驱逐速率限制在每秒 --node-eviction-rate 个(默认为 0.1)。 这表示它每 10 秒钟内至多从一个节点驱逐 Pods。

当一个可用区域Availability Zone中的节点变为不健康时节点的驱逐行为将发生改变。 节点控制器会同时检查可用区域中不健康NodeReady 状态为 ConditionUnknown 或 ConditionFalse 的节点的百分比。如果不健康节点的比例超过 --unhealthy-zone-threshold (默认为 0.55 驱逐速率将会降低:如果集群较小(意即小于等于 --large-cluster-size-threshold 个节点 - 默认为 50 驱逐操作将会停止,否则驱逐速率将降为每秒 --secondary-node-eviction-rate 个(默认为 0.01)。 在单个可用区域实施这些策略的原因是当一个可用区域可能从控制面脱离时其它可用区域可能仍然保持连接。 如果你的集群没有跨越云服务商的多个可用区域,那(整个集群)就只有一个可用区域。

跨多个可用区域部署你的节点的一个关键原因是当某个可用区域整体出现故障时,工作负载可以转移到健康的可用区域。 因此,如果一个可用区域中的所有节点都不健康时,节点控制器会以正常的速率 --node-eviction-rate 进行驱逐操作。 在所有的可用区域都不健康(也即集群中没有健康节点)的极端情况下,节点控制器将假设控制面节点的连接出了某些问题, 它将停止所有驱逐动作直到一些连接恢复。

从 Kubernetes 1.6 开始NodeController 还负责驱逐运行在拥有 NoExecute 污点的节点上的 Pods如果这些 Pods 没有容忍这些污点。 此外,作为一个默认禁用的 alpha 特性NodeController 还负责根据节点故障(例如节点不可访问或没有就绪)为其添加污点。 请查看这个文档了解 NoExecute 污点 和这个 alpha 特性。

从版本 1.8 开始,可以使节点控制器负责创建代表节点条件的污点。这是版本 1.8 的 Alpha 功能。

节点自注册

当 kubelet 标志 --register-node 为 true (默认)时,它会尝试向 API 服务注册自己。 这是首选模式,被绝大多数发行版选用。

对于自注册模式kubelet 使用下列参数启动:

  • --kubeconfig - 用于向 apiserver 表明身份的凭据路径。
  • --cloud-provider - 如何从云服务商读取关于自己的元数据。
  • --register-node - 自动向 API 服务注册。
  • --register-with-taints - 使用污点列表(逗号分隔的 <key>=<value>:<effect>)注册节点。当 register-node 为 false 时无效。
  • --node-ip - 节点 IP 地址。
  • --node-labels - 在集群中注册节点时要添加的标签(请参阅 NodeRestriction 准入插件 在 1.13+ 中实施的标签限制)。
  • --node-status-update-frequency - 指定 kubelet 向控制面组件发送状态的频率。

启用节点授权模式NodeRestriction 准入插件时, 仅授权 kubelet 创建或修改其自己的节点资源。

手动节点管理

集群管理员可以创建及修改节点对象。

如果管理员希望手动创建节点对象,请设置 kubelet 标志 --register-node=false

管理员可以修改节点资源(忽略 --register-node 设置)。修改包括在节点上设置标签及 标记它为不可调度。

节点上的标签可以和 Pods 的节点选择算符一起使用来控制调度,例如限制某 pod 只能在符合要求的节点子集上运行。

如果标记节点为不可调度的unschedulable将阻止新 Pods 调度到该节点之上,但不会影响任何已经在其上的 Pods。 这是重启节点等操作之前的一个有用的准备步骤。例如,要标记一个节点为不可调度的,执行以下命令:

kubectl cordon $NODENAME

{{< note >}} 请注意,被 DaemonSet 控制器创建的 Pods 将忽略 Kubernetes 调度器,且不会遵照节点上不可调度的属性。 这个假设基于守护程序属于节点机器,即使在准备重启而隔离应用的时候。 {{< /note >}}

节点容量

节点的容量cpu 数量和内存容量)是节点对象的一部分。 通常情况下,在创建节点对象时,它们会注册自己并报告自己的容量。 如果你正在执行手动节点管理,那么你需要在添加节点时手动设置节点容量。

Kubernetes 调度器保证一个节点上有足够的资源供其上的所有 Pods 使用。 它会检查节点上所有容器的请求的总和不会超过节点的容量。 这包括由 kubelet 启动的所有容器,但不包括由容器运行时 直接启动的容器,也不包括在容器外部运行的任何进程。

如果要为非 Pod 进程显式保留资源。请参考为系统守护程序保留资源教程。

节点拓扑

{{< feature-state state="alpha" >}}

如果启用了 TopologyManager 特性门控 则 kubelet 可以在做出资源分配决策时使用拓扑提示。

API 对象

节点是 Kubernetes REST API 的顶级资源。 更多关于 API 对象的细节可以在这里找到:[节点 API 对象](/docs/reference/generated/kubernetes-api/{{< param "version" >}}/#node-v1-core)。

{{% heading "whatsnext" %}}