一、容器生命週期
- Init C(初始化容器)只是用於 Pod 初始化的,不會一直隨著 Pod 生命週期存在,Init C 在初始化完成之後就會死亡。
- 一個 Pod 可以有多個 Init C,也可以不需要 Init C。
- Init C 是依次執行的,第一個執行成功後才可以執行下一個 Init C,不能同時執行。
Main C 退出後 Pod 生命週期就會結束,Init C 正常退出後 Pod 生命週期並不會結束,但是 Init C 不是正常退出(返回0)的話,是不會執行到 Main C 這一步的。
如果 Pod 的 Init 容器失敗,Kubernetes 會不斷的重啟該 Pod,直到 Init 容器成功為止。
如果 Pod 對應的 restartPolicy 為 Never,它不會重新啟動。
二、Init 容器
Pod 能夠具有多個容器,應用執行在容器裡面,但是它也可能有一個或多個先於應用容器啟動的 Init 容器。
Init 容器與普通容器非常相似,除了如下兩點:
- Init 容器總是執行到成功完成為止;
- 每個 Init 容器都必須在下一個Init容器啟動之前成功完成。
Init 容器使用案例
Init 模板:
apiVersion: v1
kind: Pod
metadata:
name: myapp-pod # Pod 名稱
labels:
app: myapp # Pod 標籤
spec:
containers:
- name: myapp-container # Pod 裡面第一個容器名稱
image: busybox # 該容器使用的映象名稱
command: ['sh','-c','echo The app is running! && sleep 3600']
# 執行命令,輸出一句話結束後該容器休眠6分鐘
initContainers: # 對容器初始化
- name: init-myservice # 第一個初始化容器名稱
image: busybox # 初始化容器映象
command: ['sh','-c','until nslookup myservice; do echo waiting for myservice; sleep 2; done;']
# 解析 myservice ,until 當條件為真的時候退出迴圈;
# 如果解析不成功,輸出一句話,休眠2s繼續迴圈。
- name: init-mydb # 第二個初始化容器名稱
image: busybox
command: ['sh','-c','until nslookup mydb; do echo waiting for mydb; sleep 2; done;']
# 跟第一個類似,不多追溯。
建立 Pod :
READY:總共一個需要就緒的,現在就緒狀態為0;
STAUS:總共兩個初始化容器,現在一個都沒有成功;
檢視 Init 容器日誌:
解析不成功,導致 Init 容器在不斷迴圈解析:
接下來建立一個 Service 使得 Init 容器可以解析成功:
kind: Service
apiVersion: v1
metadata:
name: myservice
spec:
ports:
- protocol: TCP
port: 80
targetPort: 9376
再次檢視 Pod 狀態:
其中第一個 Init 容器已經執行成功了,由於第二個 Init 容器還是沒有解析成功,所以 Pod 就緒狀態還是為0。
接下來建立 mydb Service :
kind: Service
apiVersion: v1
metadata:
name: mydb
spec:
ports:
- protocol: TCP
port: 80
targetPort: 9377
此時再次檢視 Pod 狀態:
只有兩個 Init 容器初始化都成功了,Main C (也就是這裡的 myapp-pod)才會執行成功。
特殊說明
- 在 Pod 啟動過程中,Init 容器會按順序在 網路 和 資料卷 初始化之後啟動,每個容器必須在下一個容器啟動之前成功退出。
這裡的“網路”和“資料卷”是指 Pause 容器,真正 Pod 第一個啟動的容器不是 Init 容器,而是 Pause 容器。
-
如果由於執行時或失敗退出,將導致容器啟動失敗,它會根據 Pod 的 restartPolicy 指定的策略進行重試。然而,如果 Pod 的 restartPolicy 設定為 Always,Init 容器失敗時會使用
RestartPolicy 策略。 -
在所有的 Init 容器沒有成功之前,Pod 將不會變成 Ready 狀態。Init 容器的埠將不會在 Service 中進行聚集。正在初始化中的 Pod 處於 Pending 狀態,但應該會將 Initializing 狀態設定為 true。
-
如果 Pod 重啟,所有 Init 容器必須重新執行。
-
在 Pod 中的每個 app 和 Init 容器的名稱必須唯一,與任何其他容器共享同一個名稱,會在驗證時丟擲錯誤。
同一組 Init 容器埠是可以相同的,因為第一個 Init 容器啟動成功後就會退出,埠就不會被佔用。
以上有不恰當或者講得不對的地方,希望各位留言指正,謝謝!
站在巨人的肩膀上!