StatefulSet:
用於部署有狀態應用
有狀態應用部署須要考慮:
1 實例的獨立存儲
2 實例之間通訊地址要固定
3 前後的啓動順序
應用場景:分佈式應用(mysql,zk,etcd)
穩定的網絡ID(實例之間通訊地址要固定)實現方法:Headless Service
normal service 與 Headless Service區別html
1 #經過deploy導出一個service模板文件供後續使用 2 [root@master ~]# kubectl create deploy web --image=nginx(也能夠用kubectl apply -f deployment2.yaml) 3 deployment.apps/web created 4 [root@master ~]# 5 [root@master ~]# kubectl expose deploy web --port=80 --target-port=80 --dry-run=client -o yaml >svc.yaml 6 [root@master ~]# 7 [root@master ~]# vim svc.yaml 8 9 apiVersion: v1 10 kind: Service 11 metadata: 12 labels: 13 app: web 14 name: web 15 spec: 16 clusterIP: None #Headless Service這裏的值是None,normal service這裏是具體的IP 17 ports: #clusterIP設置爲None的由於Headless Service不須要VIP爲後端的實際 18 - port: 80 #服務提供負載均衡,由於clinet訪問service時必須訪問真實的IP地址 19 protocol: TCP 20 targetPort: 80 21 selector: 22 app: web 23 [root@master ~]# kubectl apply -f svc.yaml 24 service/web created 25 [root@master ~]# kubectl get svc 26 NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE 27 kubernetes ClusterIP 10.96.0.1 <none> 443/TCP 93d 28 web ClusterIP None <none> 80/TCP 5s 29 [root@master ~]#
Headless Service與StatefulSet的關聯node
1 [root@master ~]# cp deployment.yaml StatefulSet1.yaml 2 [root@master ~]# cat StatefulSet1.yaml 3 apiVersion: apps/v1 4 kind: StatefulSet 5 metadata: 6 name: web 7 spec: 8 serviceName: "web" #StatefulSet中必定要有serviceName,且該值是svc.yaml文件中的Service的name 9 replicas: 3 10 selector: 11 matchLabels: 12 app: web 13 template: 14 metadata: 15 labels: 16 app: web 17 spec: 18 containers: 19 - image: nginx 20 name: nginx 21 [root@master ~]# #注意:statefulset啓動的Pod名稱分別是web-0,web-1,web-2 22 [root@master ~]# kubectl get pods 23 NAME READY STATUS RESTARTS AGE 24 nfs-client-provisioner-7676dc9cfc-j4vgl 1/1 Running 3 36h 25 web-0 1/1 Running 0 24m 26 web-1 1/1 Running 0 24m 27 web-2 1/1 Running 0 24m 28 [root@master ~]
再建立一個deployment與statefulset進行對比,在busybox容器中用CoreDNS解析來查看Pod的網絡IDmysql
1 [root@master ~]# kubectl create deploy web2 --image=nginx 2 [root@master ~]# kubectl scale deploy web2 --replicas=3 3 [root@master ~]# kubectl expose deploy web2 --port=80 --target-port=80 4 [root@master ~]# kubectl get pods 5 NAME READY STATUS RESTARTS AGE 6 nfs-client-provisioner-7676dc9cfc-j4vgl 1/1 Running 3 37h 7 web-0 1/1 Running 0 72m 8 web-1 1/1 Running 0 72m 9 web-2 1/1 Running 0 72m 10 web2-6448bfd7b7-4zs2z 1/1 Running 0 8m12s 11 web2-6448bfd7b7-tsgtv 1/1 Running 0 7m5s 12 web2-6448bfd7b7-vbvsg 1/1 Running 0 7m5s 13 [root@master ~]# kubectl get svc 14 NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE 15 kubernetes ClusterIP 10.96.0.1 <none> 443/TCP 93d 16 web ClusterIP None <none> 80/TCP 13h 17 web2 ClusterIP 10.97.186.135 <none> 80/TCP 5m44s 18 [root@master ~]# 19 #在busybox容器中驗證對service的解析 20 [root@master ~]# kubectl run -it --rm --image=busybox:1.28.4 sh 21 If you don't see a command prompt, try pressing enter. 22 / # nslookup web2.default #nslookup + serviceName.namespace 23 Server: 10.96.0.10 24 Address 1: 10.96.0.10 kube-dns.kube-system.svc.cluster.local 25 26 #coreDNS對其解析的結果正是web2的ClusterIP 27 #ClusterIP A記錄格式: 28 #<service-name>.<namespace-name>.svc.cluster.loca 29 Name: web2.default 30 Address 1: 10.97.186.135 web2.default.svc.cluster.local 31 / # 32 / # 33 / # nslookup web.default 34 Server: 10.96.0.10 35 Address 1: 10.96.0.10 kube-dns.kube-system.svc.cluster.local 36 37 #由於handless service沒有ClusterIP,因此它解析的是具體Pod的地址 38 #ClusterIP=None A記錄格式: 39 #<statefulsetName-index>.<service-name> .<namespace-name>.svc.cluster.local 40 Name: web.default 41 Address 1: 10.244.2.122 web-1.web.default.svc.cluster.local 42 Address 2: 10.244.1.111 web-2.web.default.svc.cluster.local 43 Address 3: 10.244.1.110 web-0.web.default.svc.cluster.local 44 45 / # #CoreDBS解析web-0.web.default.svc.cluster.loca的IP地址 46 #能夠看到實例web-0,web-1,web-2對應的IP地址分別是 47 #10.244.2.122,10.244.1.111,10.244.1.110 48 / # nslookup web-0.web.default.svc.cluster.local 49 Server: 10.96.0.10 50 Address 1: 10.96.0.10 kube-dns.kube-system.svc.cluster.local 51 52 Name: web-0.web.default.svc.cluster.local 53 Address 1: 10.244.1.110 web-0.web.default.svc.cluster.local 54 / # 55 / # nslookup web-1.web.default.svc.cluster.local 56 Server: 10.96.0.10 57 Address 1: 10.96.0.10 kube-dns.kube-system.svc.cluster.local 58 59 Name: web-1.web.default.svc.cluster.local 60 Address 1: 10.244.2.122 web-1.web.default.svc.cluster.local 61 / #
穩定存儲(實例的獨立存儲)的實現方法nginx
StatefulSet的存儲卷使用VolumeClaimTemplate建立,稱爲卷申請模板,當StatefulSet使用VolumeClaimTemplate
建立一個PersistentVolume時,一樣也會爲每一個Pod分配並建立一個編號的PVC。web
1 #刪除上個實驗的pod 2 [root@master ~]# kubectl delete -f StatefulSet1.yaml 3 statefulset.apps "web" deleted 4 [root@master ~]# kubectl get pods 5 NAME READY STATUS RESTARTS AGE 6 nfs-client-provisioner-7676dc9cfc-j4vgl 1/1 Running 3 44h 7 sh 1/1 Running 0 6h16m 8 web2-6448bfd7b7-4zs2z 1/1 Running 0 6h39m 9 web2-6448bfd7b7-tsgtv 1/1 Running 0 6h37m 10 web2-6448bfd7b7-vbvsg 1/1 Running 0 6h37m 11 [root@master ~]# 12 #再次編輯一個statefulset文件,在文件中使用VolumeClaimTemplate 13 #爲每一個Pod提供一個PV 14 [root@master ~]# cp StatefulSet1.yaml StatefulSet2.yaml 15 [root@master ~]# vim StatefulSet2.yaml 16 apiVersion: apps/v1 17 kind: StatefulSet 18 metadata: 19 name: web 20 spec: 21 serviceName: "web" 22 replicas: 3 23 selector: 24 matchLabels: 25 app: web 26 template: 27 metadata: 28 labels: 29 app: web 30 spec: 31 containers: 32 - image: nginx 33 name: nginx 34 volumeMounts: 35 - name: www #nginx容器中掛載的卷名稱是www,要與volumeClaimTemplates中的name一致 36 mountPath: /usr/share/nginx/html 37 volumeClaimTemplates: 38 - metadata: 39 name: www 40 spec: 41 accessModes: [ "ReadWriteOnce" ] 42 storageClassName: "managed-nfs-storage" 43 resources: 44 requests: 45 storage: 1Gi 46 [root@master ~]# 47 48 [root@master ~]# kubectl apply -f StatefulSet2.yaml 49 statefulset.apps/web created 50 [root@master ~]# 51 [root@master ~]# kubectl get pods 52 NAME READY STATUS RESTARTS AGE 53 nfs-client-provisioner-7676dc9cfc-j4vgl 1/1 Running 3 44h 54 sh 1/1 Running 0 6h19m 55 web-0 1/1 Running 0 63s 56 web-1 1/1 Running 0 59s 57 web-2 1/1 Running 0 55s 58 web2-6448bfd7b7-4zs2z 1/1 Running 0 6h42m 59 web2-6448bfd7b7-tsgtv 1/1 Running 0 6h41m 60 web2-6448bfd7b7-vbvsg 1/1 Running 0 6h41m 61 [root@master ~]# 62 #卷申請模板爲每一個pod建立一個pvc,且每一個pvc關聯一個pv 63 [root@master ~]# kubectl get pv,pvc 64 NAME CAPACITY ACCESS MODES RECLAIM POLICY STATUS CLAIM STORAGECLASS REASON AGE 65 persistentvolume/pv01 5Gi RWX Retain Available 2d1h 66 persistentvolume/pv02 10Gi RWX Retain Released default/my-pvc 2d1h 67 persistentvolume/pv03 20Gi RWX Retain Available 2d1h 68 persistentvolume/pvc-7b3836a2-15a8-4584-9635-7be34b304a94 1Gi RWO Delete Bound default/www-web-0 managed-nfs-storage 115s 69 persistentvolume/pvc-ae5f09e6-493e-47d5-8268-0a850c892890 1Gi RWO Delete Bound default/www-web-1 managed-nfs-storage 111s 70 persistentvolume/pvc-eda3b8d3-56b3-4a33-a5d3-3af39e5f76ad 1Gi RWO Delete Bound default/www-web-2 managed-nfs-storage 107s 71 72 NAME STATUS VOLUME CAPACITY ACCESS MODES STORAGECLASS AGE 73 persistentvolumeclaim/www-web-0 Bound pvc-7b3836a2-15a8-4584-9635-7be34b304a94 1Gi RWO managed-nfs-storage 116s 74 persistentvolumeclaim/www-web-1 Bound pvc-ae5f09e6-493e-47d5-8268-0a850c892890 1Gi RWO managed-nfs-storage 111s 75 persistentvolumeclaim/www-web-2 Bound pvc-eda3b8d3-56b3-4a33-a5d3-3af39e5f76ad 1Gi RWO managed-nfs-storage 107s 76 [root@master ~]# 77 78 #在nfs服務器上查看Pv的物理目錄 79 [root@node2 ~]# cd /ifs/kubernetes/ 80 [root@node2 kubernetes]# ll 81 total 8 82 drwxrwxrwx 2 root root 21 Feb 16 15:54 archived-default-my-pvc2-pvc-cfa1317b-67eb-4e36-8c4d-75e7c075e2a9 83 -rw-r--r-- 1 root root 9 Feb 13 20:59 a.txt 84 drwxrwxrwx 2 root root 6 Feb 17 18:08 default-www-web-0-pvc-7b3836a2-15a8-4584-9635-7be34b304a94 85 drwxrwxrwx 2 root root 6 Feb 17 18:08 default-www-web-1-pvc-ae5f09e6-493e-47d5-8268-0a850c892890 86 drwxrwxrwx 2 root root 6 Feb 17 18:08 default-www-web-2-pvc-eda3b8d3-56b3-4a33-a5d3-3af39e5f76ad 87 -rw-r--r-- 1 root root 18 Feb 13 23:10 index.html 88 drwxr-xr-x 2 root root 6 Feb 15 16:43 pv01 89 drwxr-xr-x 2 root root 6 Feb 15 16:43 pv02 90 drwxr-xr-x 2 root root 6 Feb 15 16:43 pv03 91 [root@node2 kubernetes]#