data:image/s3,"s3://crabby-images/562d7/562d71444d3c946ba56122f0ab6aa3770f0c43b7" alt="GKE上同一组pod同时支持多种Service模式"
GKE 是 Google Cloud 的 Kubernetes 的服务。其架构如下:
data:image/s3,"s3://crabby-images/9001d/9001d3a4c7b25df1295cb7538e6ff56275da7c7b" alt=""
Master Node 是 Google 免费提供的,用户自己创建 Node Pool,不同的 Node Pool 可以采用不同的硬件配置,比如带 GPU 或不带 GPU。
在 GKE 上,创建不同的对外的服务接口,相应的将自动生成不同的服务:
如果创建一个 ingress,系统将自动创建一个 http(s)的负载均衡或 L7 的 internal LB;
如果创建一个 LoadBalancer 类型的 Service,系统将自动创建一个 TCP 的负载均衡;
如果创建一个 LoadBalancer 类型的内网地址的 service,系统将自动创建一个 TCP 的 Internal loadbalancer;
如果创建一个 nodeport 类型的 Service,不会自动生成 loadBalancer
本文将介绍这几种 service 如何创建,并特别介绍如何同时支持 ingress 和内网地址的 service。
创建 GKE cluster
创建 cluster
通过界面创建 GKE 的 cluster:
data:image/s3,"s3://crabby-images/a60cb/a60cb15262f866f06cfac7c53741bd3ee5f7e1be" alt=""
Cluster 创建成功:
data:image/s3,"s3://crabby-images/6128e/6128e8b6d69245e621265570238bbe358670e8d4" alt=""
kubectl 与 cluster 连接
在界面上点击 connect
data:image/s3,"s3://crabby-images/7e8ae/7e8aef04413b81d37028d03591e3f617af6417f4" alt=""
data:image/s3,"s3://crabby-images/854fc/854fcf3ccb0a7b0d5531a3d4e86291da2852f7aa" alt=""
复制 gcloud 命令
并在 cloud shell 中运行:
data:image/s3,"s3://crabby-images/a17a5/a17a5561b3c8eaebea10ac0956641fe93b43f46c" alt=""
连接成功,可以通过 kubectl 控制 cluster:
data:image/s3,"s3://crabby-images/41ec8/41ec89c62e627079ad32e6b8b67e8698ff918d0b" alt=""
二 部署 workload
通过 kubectl 命令创建 3 个 pod nginx 的 deployment:
在 console 上可以看到:
data:image/s3,"s3://crabby-images/c9aa5/c9aa5e52cc670f46760e32ea8cd1b3475b8b5dda" alt=""
通过 kubectl 查看:
data:image/s3,"s3://crabby-images/03fd3/03fd36c4f22d730e7208fcd1ed4bec56563cd0bf" alt=""
cluster 网络
pod IP 地址段
查看 cluster 的详情:
data:image/s3,"s3://crabby-images/ddcb8/ddcb8b2de86803cefbd21b026bcc9b96e9282f61" alt=""
因为 Google 的 GKE 采用的是 per pod per IP 的模式,Pod 的 IP 地址在 VPC 内:
data:image/s3,"s3://crabby-images/1a7a8/1a7a889b9d03ad0d9382437fbbf1c211692b2c07" alt=""
这段地址是在创建 cluster 时自动创建的,专门用于 pods 的 IP 地址。另外在 VPC 内还有一段 IP 地址用于将来 Service 使用,比如 cluster IP。
node IP 地址
data:image/s3,"s3://crabby-images/8105c/8105cc8a9a158324d06e3c8eb99d01e1339375be" alt=""
查看 node 的 IP:
可以看到,除标准段 primary Internal IP 外,这个 node 有一个 alias IP,是在 pods IP 内的一个 24 位掩码的地址段。在这个 node 上的 pod 将会使用这段地址。这样,每个 node 会有一段 IP 地址。这种设计和 K8s 网络的 flannel 类似:
data:image/s3,"s3://crabby-images/87209/87209b534b0f5c50f75c7b935172383cf5abbc6b" alt=""
GKE 中,这段 Alias IP 用于 Pod 间通讯,也可以用于和 GCE 的通讯。如上图,Node2 的 IP 地址为 10.1.1.22,Pod 的 IP 地址段是 10.40.2.0/24。此 Node 上的 Pod 地址都在这个 Alias IP 段内。GCE 10.100.1.10 与 Node2 的 Pod 进行通讯,可以直接和 10.40.2.10 或 10.40.2.11 直接通讯。跨 Node 的 Pod 间通讯,也直接用 Pod IP 进行通讯,比如 10.40.1.10 <-> 10.40.2.10。
pod 的 IP
查看 pod 信息可以看到 3 个 pod,运行在不同的 node 上,并且 IP 地址是 10.40.0.0/24,10.40.1.0/24, 10.40.2.0/24 的地址段。印证了我们前面观察到的 node ip 地址段。
data:image/s3,"s3://crabby-images/b6ba3/b6ba3c51aca02fa6f58774216a20bcd99b7d8552" alt=""
在同一个网段的 VM 去 ping 和 curl 一个 pod,发现可以直接访问:
data:image/s3,"s3://crabby-images/23198/2319840436d93c185c20176ec36456e34122eae5" alt=""
如前所述,这样的设计解决了在 K8s 集群中,跨 node 的 pod 通讯问题,也解决了非 cluster 内 IP 和 pod 通讯的问题。
创建 Load Balancer 类型的 service
创建 Service
在 Google Cloud console 中,
data:image/s3,"s3://crabby-images/57ef2/57ef28c876ca8ab2ede4d77153d083f21da59e01" alt=""
在 workload 界面中选择 expose
data:image/s3,"s3://crabby-images/338bd/338bd72df40ceecbad58a9980400ebb4191aff64" alt=""
填入相应的参数,点击 expose:
data:image/s3,"s3://crabby-images/36332/363329e9309caa92972f79197f760f1dfc1820d8" alt=""
由于类型选择的是 loadbalancer,GCP 会自动创建一个 TCP 的 loadbalancer。
Service 详情
首先查看 service:
data:image/s3,"s3://crabby-images/7da4d/7da4d48d320ebb53f5b4c360021246f3478c847b" alt=""
看到 service 已经创建成功,类型是 load balancer 的。
负载均衡详情
再查看 loadbalancer:
data:image/s3,"s3://crabby-images/8caf7/8caf7f33f10abd49bdeeea9e12ed6180358eb6e5" alt=""
可以看到一个 TCP 的 load balancer 已经创建成功。
后端采用的是添加 Instance 的方式:
data:image/s3,"s3://crabby-images/e0baa/e0baa7fa3a14220cc35811fd784475a646b160aa" alt=""
前端生成一个公网 IP 地址:
data:image/s3,"s3://crabby-images/e8b8b/e8b8b7215d4cb93433779748ce5d8a2e154b87d6" alt=""
通过这个 IP 地址,可以访问到访问:
data:image/s3,"s3://crabby-images/1b811/1b81132cee66d6e74a05977200130577276db9da" alt=""
创建内部 IP 的 service
通过 yaml 文件创建
在 cloud shell 中创建一个 yaml 文件:
data:image/s3,"s3://crabby-images/0a19b/0a19bc0b100f27b7606516810c7f8219d9833050" alt=""
apiVersion: v1
通过 kubectl 创建这个内部 IP 的 service:
service 详情
data:image/s3,"s3://crabby-images/84674/84674e4f5bcee1db1513419cc0a3b59fdeb7a761" alt=""
在 console 上可以看到已经创建成功。
详细信息中,可以看到:
data:image/s3,"s3://crabby-images/1e3cc/1e3cca53a3612eddcfe2ff0c5633b74297b46601" alt=""
内部 IP 是 10.1.1.27,这个地址是这个网段的 primary IP 地址段,Cluster IP 在 cluster 中定义的一致,后端有 3 个 pod,服务端口是 80,node port 是 30660,pod 内的 port 是 80.
ILB 详情
查看 ilb 信息:
data:image/s3,"s3://crabby-images/73f24/73f24b34f419c728ea8d7252e19b06f8e11fc3bd" alt=""
在详细信息中,可以看到 backend 信息:
data:image/s3,"s3://crabby-images/64995/649951dc98d6db5af4b15bfe98943a89cbaa6ebf" alt=""
ILB 的后端采用 instance group 的形式。
ILB 的前端:
data:image/s3,"s3://crabby-images/6b58c/6b58cc90ef823c60d7c569ed4a3eec9322173bb5" alt=""
创建 ingress
创建基于 node port 的 service
创建 yaml 文件:
通过 kubectl 创建:
nodeport service 详情
data:image/s3,"s3://crabby-images/26837/268378b6b1c794b7a05a8e8284ec2007938dc33b" alt=""
详情如下:
data:image/s3,"s3://crabby-images/fb61e/fb61ec9d188a32d03540f2a9306440466c6ee20d" alt=""
创建 ingress
创建 ingress 的 yaml 文件:
创建 ingress:
Ingress 详情
可以看到 ingress 创建成功:
data:image/s3,"s3://crabby-images/be146/be146c7613f96c0d0320b431d17f76384717153e" alt=""
详细内容:
data:image/s3,"s3://crabby-images/a3ea7/a3ea7156afe68b1e7f2d8898b4c6a67ec64d44f5" alt=""
http 负载均衡详情
可以看到 http 负载均衡已经创建成功:
data:image/s3,"s3://crabby-images/40119/4011968a46bceff02c3e74b388035647f5431444" alt=""
后端情况,有两个 backend,但 instance group 相同:
data:image/s3,"s3://crabby-images/7ad64/7ad64a1d09436033b44bf596311790bd30eb933f" alt=""
Load balancer mode 采用的是 rate 模式:
data:image/s3,"s3://crabby-images/80a53/80a535c4917fc77f3973bc32716dacaeef7e9550" alt=""
Rule 有 3 条,分布引用这两个 backend service:
data:image/s3,"s3://crabby-images/e480d/e480de0020bc424dfef6c69f9f053923494d341b" alt=""
前端是有公网 IP 的 HTTP 配置:
data:image/s3,"s3://crabby-images/dab5b/dab5bfe8a740fdd9ca0cd5f19163a70462e0835e" alt=""
通过这个公网 IP,同样可以访问到服务:
data:image/s3,"s3://crabby-images/0f969/0f9697ea1b730877f08e52be2f70419fbc9113fc" alt=""
注意事项
注意到在 ingress 的 http 负载均衡中,load balancer 模式采用的是 rate 模式。
在实际操作中,如果先创建 ingress,有可能会采用 utilization 模式,而 ILB 的 backend 只能采用 rate 模式,此时再创建内部 IP 的 service 时,会出错:
data:image/s3,"s3://crabby-images/ea468/ea468adb0f5644edf52634927308f2085d92d0d7" alt=""
查看 HTTP 负载均衡情况:
data:image/s3,"s3://crabby-images/bcde9/bcde9401d019e2847461abf7d140e46981b5b469" alt=""
这个是 utilization 模式。造成不能创建 ILB。
目前,建议先创建内部 IP 的 service,再创建 ingress,这样可以避免这个问题。
总结
通过介绍如何创建 GKE 上各种 Service 类型,介绍了 Cluster 的创建、GKE 的网络组成、Ingress、ILB 模式的 Service,nodeport 模式的 service 的 Yaml 文件。以及在需要同时支持 ingress 和 ILB 模式 Service 是要注意的问题。
评论