网络异常排错

    说到 Kubernetes 的网络,其实无非就是以下三种情况之一

    • Pod 访问容器外部网络
    • 从容器外部访问 Pod 网络
    • Pod 之间相互访问

    当然,以上每种情况还都分别包括本地访问和跨主机访问两种场景,并且一般情况下都是通过 Service 间接访问 Pod。

    排查网络问题基本上也是从这几种情况出发,定位出具体的网络异常点,再进而寻找解决方法。网络异常可能的原因比较多,常见的有

    • CNI 网络插件配置错误,导致多主机网络不通,比如
      • IP 网段与现有网络冲突
      • 插件使用了底层网络不支持的协议
      • 忘记开启 IP 转发等
        • sysctl net.bridge.bridge-nf-call-iptables
    • Pod 网络路由丢失,比如
      • kubenet 要求网络中有 podCIDR 到主机 IP 地址的路由,这些路由如果没有正确配置会导致 Pod 网络通信等问题
      • 在公有云平台上,kube-controller-manager 会自动为所有 Node 配置路由,但如果配置不当(如认证授权失败、超出配额等),也有可能导致无法配置路由
    • Service NodePort 和 health probe 端口冲突
      • 在 1.10.4 版本之前的集群中,多个不同的 Service 之间的 NodePort 和 health probe 端口有可能会有重合 (已经在 kubernetes#64468 修复)
    • 主机内或者云平台的安全组、防火墙或者安全策略等阻止了 Pod 网络,比如
      • 非 Kubernetes 管理的 iptables 规则禁止了 Pod 网络
      • 公有云平台的安全组禁止了 Pod 网络(注意 Pod 网络有可能与 Node 网络不在同一个网段)
      • 交换机或者路由器的 ACL 禁止了 Pod 网络

    Flannel 网络插件非常容易部署,只要一条命令即可

    然而,部署完成后,Flannel Pod 有可能会碰到初始化失败的错误

    1. $ kubectl -n kube-system get pod
    2. NAME READY STATUS RESTARTS AGE
    3. kube-flannel-ds-ckfdc 0/1 Init:CrashLoopBackOff 4 2m
    4. kube-flannel-ds-jpp96 0/1 Init:CrashLoopBackOff 4 2m

    查看日志会发现

    1. $ kubectl -n kube-system logs kube-flannel-ds-jpp96 -c install-cni
    2. cp: can't create '/etc/cni/net.d/10-flannel.conflist': Permission denied

    这一般是由于 SELinux 开启导致的,关闭 SELinux 既可解决。有两种方法:

    • 修改 /etc/selinux/config 文件方法:SELINUX=disabled
    • 通过命令临时修改(重启会丢失):setenforce 0

    Pod 无法分配 IP

    Pod 一直处于 ContainerCreating 状态,查看事件发现网络插件无法为其分配 IP:

    1. Normal SandboxChanged 5m (x74 over 8m) kubelet, k8s-agentpool-66825246-0 Pod sandbox changed, it will be killed and re-created.
    2. Warning FailedCreatePodSandBox 21s (x204 over 8m) kubelet, k8s-agentpool-66825246-0 Failed create pod sandbox: rpc error: code = Unknown desc = NetworkPlugin cni failed to set up pod "deployment-azuredisk6-56d8dcb746-487td_default" network: Failed to allocate address: Failed to delegate: Failed to allocate address: No available addresses

    查看网络插件的 IP 分配情况,进一步发现 IP 地址确实已经全部分配完,但真正处于 Running 状态的 Pod 数却很少:

    1. # 详细路径取决于具体的网络插件,当使用 host-local IPAM 插件时,路径位于 /var/lib/cni/networks 下面
    2. $ cd /var/lib/cni/networks/kubenet
    3. $ ls -al|wc -l
    4. 258
    5. $ docker ps | grep POD | wc -l
    6. 7

    这有两种可能的原因

    • 网络插件本身的问题,Pod 停止后其 IP 未释放
    • Pod 重新创建的速度比 Kubelet 调用 CNI 插件回收网络(垃圾回收时删除已停止 Pod 前会先调用 CNI 清理网络)的速度快

    对第一个问题,最好联系插件开发者询问修复方法或者临时性的解决方法。当然,如果对网络插件的工作原理很熟悉的话,也可以考虑手动释放未使用的 IP 地址,比如:

    • 停止 Kubelet
    • 找到 IPAM 插件保存已分配 IP 地址的文件,比如 /var/lib/cni/networks/cbr0(flannel)或者 /var/run/azure-vnet-ipam.json(Azure CNI)等
    • 查询容器已用的 IP 地址,比如 kubectl get pod -o wide --all-namespaces | grep <node-name>
    • 对比两个列表,从 IPAM 文件中删除未使用的 IP 地址,并手动删除相关的虚拟网卡和网络命名空间(如果有的话)
    • 重启启动 Kubelet
    1. # Take kubenet for example to delete the unused IPs
    2. $ for hash in $(tail -n +1 * | grep '^[A-Za-z0-9]*$' | cut -c 1-8); do if [ -z $(docker ps -a | grep $hash | awk '{print $1}') ]; then grep -ilr $hash ./; fi; done | xargs rm

    而第二个问题则可以给 Kubelet 配置更快的垃圾回收,如

    1. --minimum-container-ttl-duration=15s
    2. --maximum-dead-containers-per-container=1
    3. --maximum-dead-containers=100

    Pod 无法解析 DNS

    如果 Node 上安装的 Docker 版本大于 1.12,那么 Docker 会把默认的 iptables FORWARD 策略改为 DROP。这会引发 Pod 网络访问的问题。解决方法则在每个 Node 上面运行 iptables -P FORWARD ACCEPT,比如

    除此之外,还有很多其他原因导致 DNS 无法解析:

    (1)DNS 无法解析也有可能是 kube-dns 服务异常导致的,可以通过下面的命令来检查 kube-dns 是否处于正常运行状态

    1. $ kubectl get pods --namespace=kube-system -l k8s-app=kube-dns
    2. NAME READY STATUS RESTARTS AGE
    3. ...
    4. kube-dns-v19-ezo1y 3/3 Running 0 1h

    如果 kube-dns 处于 CrashLoopBackOff 状态,那么可以参考 来查看具体排错方法。

    (2)如果 kube-dns Pod 处于正常 Running 状态,则需要进一步检查是否正确配置了 kube-dns 服务:

    1. $ kubectl get svc kube-dns --namespace=kube-system
    2. NAME CLUSTER-IP EXTERNAL-IP PORT(S) AGE
    3. kube-dns 10.0.0.10 <none> 53/UDP,53/TCP 1h
    4. $ kubectl get ep kube-dns --namespace=kube-system
    5. NAME ENDPOINTS AGE
    6. kube-dns 10.180.3.17:53,10.180.3.17:53 1h

    如果 kube-dns service 不存在,或者 endpoints 列表为空,则说明 kube-dns service 配置错误,可以重新创建 kube-dns service,比如

    1. apiVersion: v1
    2. kind: Service
    3. metadata:
    4. name: kube-dns
    5. namespace: kube-system
    6. labels:
    7. k8s-app: kube-dns
    8. kubernetes.io/cluster-service: "true"
    9. kubernetes.io/name: "KubeDNS"
    10. selector:
    11. k8s-app: kube-dns
    12. clusterIP: 10.0.0.10
    13. ports:
    14. - name: dns
    15. port: 53
    16. protocol: UDP
    17. - name: dns-tcp
    18. port: 53
    19. protocol: TCP

    (3)如果最近升级了 CoreDNS,并且使用了 CoreDNS 的 proxy 插件,那么需要注意 的版本需要将 proxy 插件替换为 forward 插件。比如:

    1. apiVersion: v1
    2. kind: ConfigMap
    3. metadata:
    4. name: coredns-custom
    5. namespace: kube-system
    6. data:
    7. test.server: |
    8. abc.com:53 {
    9. errors
    10. cache 30
    11. forward . 1.2.3.4
    12. }
    13. my.cluster.local:53 {
    14. errors
    15. cache 30
    16. forward . 2.3.4.5
    17. }
    18. azurestack.local:53 {
    19. forward . tls://1.1.1.1 tls://1.0.0.1 {
    20. tls_servername cloudflare-dns.com
    21. health_check 5s
    22. }
    23. }

    (4)如果 kube-dns Pod 和 Service 都正常,那么就需要检查 kube-proxy 是否正确为 kube-dns 配置了负载均衡的 iptables 规则。具体排查方法可以参考下面的 Service 无法访问部分。

    由于内核的一个 BUG,连接跟踪模块会发生竞争,导致 DNS 解析缓慢。社区跟踪 Issue 为 。

    临时解决方法:为容器配置 options single-request-reopen,避免相同五元组的并发 DNS 请求:

    1. lifecycle:
    2. postStart:
    3. exec:
    4. command:
    5. - /bin/sh
    6. - -c
    7. - "/bin/echo 'options single-request-reopen' >> /etc/resolv.conf"

    或者为 Pod 配置 dnsConfig:

    1. template:
    2. spec:
    3. dnsConfig:
    4. - name: single-request-reopen

    修复方法:升级内核并保证包含以下两个补丁

    1. (included since kernel v5.0)
    2. netfilter: nf_conntrack: resolve clash for matching conntracks (included since kernel v4.19)
    • Kube-dns 和 CoreDNS 同时存在时也会有问题,只保留一个即可。
    • kube-dns 或者 CoreDNS 的资源限制太小时会导致 DNS 解析缓慢,这时候需要增大资源限制。
    • 配置 DNS 选项 use-vc,强制使用 TCP 协议发送 DNS 查询。
    • 在每个节点上运行一个 DNS 缓存服务,然后把所有容器的 DNS nameserver 都指向该缓存。

    推荐部署 Nodelocal DNS Cache 扩展来解决这个问题,同时也提升 DNS 解析的性能。 Nodelocal DNS Cache 的部署步骤请参考 。

    Service 无法访问

    访问 Service ClusterIP 失败时,可以首先确认是否有对应的 Endpoints

    如果该列表为空,则有可能是该 Service 的 LabelSelector 配置错误,可以用下面的方法确认一下

    1. # 查询 Service 的 LabelSelector
    2. # 查询匹配 LabelSelector 的 Pod
    3. kubectl get pods -l key1=value1,key2=value2

    如果 Endpoints 正常,可以进一步检查

    • Pod 的 containerPort 与 Service 的 containerPort 是否对应
    • 直接访问 podIP:containerPort 是否正常

    再进一步,即使上述配置都正确无误,还有其他的原因会导致 Service 无法访问,比如

    • Pod 内的容器有可能未正常运行或者没有监听在指定的 containerPort 上
    • CNI 网络或主机路由异常也会导致类似的问题
    • kube-proxy 服务有可能未启动或者未正确配置相应的 iptables 规则,比如正常情况下名为 hostnames 的服务会配置以下 iptables 规则
    1. $ iptables-save | grep hostnames
    2. -A KUBE-SEP-57KPRZ3JQVENLNBR -s 10.244.3.6/32 -m comment --comment "default/hostnames:" -j MARK --set-xmark 0x00004000/0x00004000
    3. -A KUBE-SEP-57KPRZ3JQVENLNBR -p tcp -m comment --comment "default/hostnames:" -m tcp -j DNAT --to-destination 10.244.3.6:9376
    4. -A KUBE-SEP-WNBA2IHDGP2BOBGZ -s 10.244.1.7/32 -m comment --comment "default/hostnames:" -j MARK --set-xmark 0x00004000/0x00004000
    5. -A KUBE-SEP-WNBA2IHDGP2BOBGZ -p tcp -m comment --comment "default/hostnames:" -m tcp -j DNAT --to-destination 10.244.1.7:9376
    6. -A KUBE-SEP-X3P2623AGDH6CDF3 -s 10.244.2.3/32 -m comment --comment "default/hostnames:" -j MARK --set-xmark 0x00004000/0x00004000
    7. -A KUBE-SEP-X3P2623AGDH6CDF3 -p tcp -m comment --comment "default/hostnames:" -m tcp -j DNAT --to-destination 10.244.2.3:9376
    8. -A KUBE-SERVICES -d 10.0.1.175/32 -p tcp -m comment --comment "default/hostnames: cluster IP" -m tcp --dport 80 -j KUBE-SVC-NWV5X2332I4OT4T3
    9. -A KUBE-SVC-NWV5X2332I4OT4T3 -m comment --comment "default/hostnames:" -m statistic --mode random --probability 0.33332999982 -j KUBE-SEP-WNBA2IHDGP2BOBGZ
    10. -A KUBE-SVC-NWV5X2332I4OT4T3 -m comment --comment "default/hostnames:" -m statistic --mode random --probability 0.50000000000 -j KUBE-SEP-X3P2623AGDH6CDF3
    11. -A KUBE-SVC-NWV5X2332I4OT4T3 -m comment --comment "default/hostnames:" -j KUBE-SEP-57KPRZ3JQVENLNBR

    Pod 无法通过 Service 访问自己

    这通常是 hairpin 配置错误导致的,可以通过 Kubelet 的 --hairpin-mode 选项配置,可选参数包括 “promiscuous-bridge”、”hairpin-veth” 和 “none”(默认为”promiscuous-bridge”)。

    对于 hairpin-veth 模式,可以通过以下命令来确认是否生效

    1. $ for intf in /sys/devices/virtual/net/cbr0/brif/*; do cat $intf/hairpin_mode; done
    2. 1
    3. 1
    4. 1
    5. 1

    而对于 promiscuous-bridge 模式,可以通过以下命令来确认是否生效

    1. $ ifconfig cbr0 |grep PROMISC
    2. UP BROADCAST RUNNING PROMISC MULTICAST MTU:1460 Metric:1

    很多扩展服务需要访问 Kubernetes API 查询需要的数据(比如 kube-dns、Operator 等)。通常在 Kubernetes API 无法访问时,可以首先通过下面的命令验证 Kubernetes API 是正常的:

    1. $ kubectl run curl --image=appropriate/curl -i -t --restart=Never --command -- sh
    2. If you don't see a command prompt, try pressing enter.
    3. / #
    4. / # KUBE_TOKEN=$(cat /var/run/secrets/kubernetes.io/serviceaccount/token)
    5. / # curl -sSk -H "Authorization: Bearer $KUBE_TOKEN" https://$KUBERNETES_SERVICE_HOST:$KUBERNETES_SERVICE_PORT/api/v1/namespaces/default/pods
    6. {
    7. "kind": "PodList",
    8. "apiVersion": "v1",
    9. "metadata": {
    10. "selfLink": "/api/v1/namespaces/default/pods",
    11. "resourceVersion": "2285"
    12. },
    13. "items": [
    14. ...
    15. ]
    16. }

    如果出现超时错误,则需要进一步确认名为 kubernetes 的服务以及 endpoints 列表是正常的:

    1. $ kubectl get service kubernetes
    2. NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
    3. kubernetes ClusterIP 10.96.0.1 <none> 443/TCP 25m
    4. $ kubectl get endpoints kubernetes
    5. NAME ENDPOINTS AGE
    6. kubernetes 172.17.0.62:6443 25m

    然后可以直接访问 endpoints 查看 kube-apiserver 是否可以正常访问。无法访问时通常说明 kube-apiserver 未正常启动,或者有防火墙规则阻止了访问。

    但如果出现了 403 - Forbidden 错误,则说明 Kubernetes 集群开启了访问授权控制(如 RBAC),此时就需要给 Pod 所用的 ServiceAccount 创建角色和角色绑定授权访问所需要的资源。比如 CoreDNS 就需要创建以下 ServiceAccount 以及角色绑定:

    内核导致的问题

    • 。注意, Kubernetes 暂时没有为 SNAT 设置 选项,如果碰到这个问题可以参考这里 配置。

    参考文档