環境
- kubernetes 1.20.2
- Spring Boot 2.5.0-M1
目標
在前面,我們通過 Service 提供的環境變量,只要知道了服務的名稱,就可以訪問到該服務,但是這樣還存在一個問題。
就是如果 Pod 在 Service 之前創建,那么 Service 的信息是不會自動注入進去的。
問題復現
查看當前 Pod
[root@master kubernetes]# kubectl get pod -o wide
NAME READY STATUS RESTARTS AGE IP NODE NOMINATED NODE READINESS GATES
rc-demo-46dq7 1/1 Running 0 15s 10.244.1.113 node1 <none> <none>
rc-demo-dxvlv 1/1 Running 0 15s 10.244.1.111 node1 <none> <none>
rc-demo-j48b8 1/1 Running 0 15s 10.244.1.112 node1 <none> <none>
查看全部環境變量
[root@master kubernetes]# kubectl exec rc-demo-j48b8 -- printenv
PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/lib/jvm/java-1.8-openjdk/jre/bin:/usr/lib/jvm/java-1.8-openjdk/bin
HOSTNAME=rc-demo-j48b8
SPRING_DATASOURCE_URL=jdbc:postgresql://${SVC_PG_SERVICE_HOST}:${SVC_PG_SERVICE_PORT}/postgres
SERVER_PORT=80
KUBERNETES_SERVICE_HOST=10.96.0.1
KUBERNETES_PORT_443_TCP=tcp://10.96.0.1:443
KUBERNETES_SERVICE_PORT=443
KUBERNETES_SERVICE_PORT_HTTPS=443
SVC_PG_SERVICE_PORT=5432
SVC_PG_PORT_5432_TCP=tcp://10.108.222.50:5432
SVC_PG_PORT_5432_TCP_PROTO=tcp
SVC_PG_PORT_5432_TCP_ADDR=10.108.222.50
KUBERNETES_PORT=tcp://10.96.0.1:443
KUBERNETES_PORT_443_TCP_PROTO=tcp
KUBERNETES_PORT_443_TCP_PORT=443
KUBERNETES_PORT_443_TCP_ADDR=10.96.0.1
SVC_PG_SERVICE_HOST=10.108.222.50
SVC_PG_PORT=tcp://10.108.222.50:5432
SVC_PG_PORT_5432_TCP_PORT=5432
LANG=C.UTF-8
JAVA_HOME=/usr/lib/jvm/java-1.8-openjdk/jre
JAVA_VERSION=8u212
JAVA_ALPINE_VERSION=8.212.04-r0
HOME=/root
新建 Service
svc.yaml
apiVersion: v1
kind: Service
metadata:
name: svc-demo
spec:
selector:
app: myapp
ports:
- port: 5432
查看重啟后狀態
[root@master kubernetes]# kubectl apply -f svc.yaml
service/svc-demo created
[root@master kubernetes]# kubectl get service
NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
kubernetes ClusterIP 10.96.0.1 <none> 443/TCP 7d
svc-demo ClusterIP 10.104.212.163 <none> 5432/TCP 4s
svc-pg ClusterIP 10.108.222.50 <none> 5432/TCP 69m
再次查看環境變量
可以看到 Service svc-demo 的環境變量並沒有注入進去,需要重啟 Pod 才可以。
[root@master kubernetes]# kubectl exec rc-demo-j48b8 -- printenv
PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/lib/jvm/java-1.8-openjdk/jre/bin:/usr/lib/jvm/java-1.8-openjdk/bin
HOSTNAME=rc-demo-j48b8
SPRING_DATASOURCE_URL=jdbc:postgresql://${SVC_PG_SERVICE_HOST}:${SVC_PG_SERVICE_PORT}/postgres
SERVER_PORT=80
KUBERNETES_SERVICE_HOST=10.96.0.1
KUBERNETES_PORT_443_TCP=tcp://10.96.0.1:443
KUBERNETES_SERVICE_PORT=443
KUBERNETES_SERVICE_PORT_HTTPS=443
SVC_PG_SERVICE_PORT=5432
SVC_PG_PORT_5432_TCP=tcp://10.108.222.50:5432
SVC_PG_PORT_5432_TCP_PROTO=tcp
SVC_PG_PORT_5432_TCP_ADDR=10.108.222.50
KUBERNETES_PORT=tcp://10.96.0.1:443
KUBERNETES_PORT_443_TCP_PROTO=tcp
KUBERNETES_PORT_443_TCP_PORT=443
KUBERNETES_PORT_443_TCP_ADDR=10.96.0.1
SVC_PG_SERVICE_HOST=10.108.222.50
SVC_PG_PORT=tcp://10.108.222.50:5432
SVC_PG_PORT_5432_TCP_PORT=5432
LANG=C.UTF-8
JAVA_HOME=/usr/lib/jvm/java-1.8-openjdk/jre
JAVA_VERSION=8u212
JAVA_ALPINE_VERSION=8.212.04-r0
HOME=/root
總結
雖然通過 Service 的環境變量訪問服務很方便,知道名字就可以。但是會存在 Pod 比 Service 先啟動,不會注入的問題。
同時修改的時候,也不會更新現有的 Pod。所以通過環境變量的方式訪問服務,可靠性方面得不到保證。