目录

1.1  Statefulset控制器:概念、原理解读

1.2  Statefulset资源清单文件编写技巧

1.3 Statefulset使用案例:部署web站点

1.4 Statefulset管理pod:扩容、缩容、更新


文档中的YAML文件配置直接复制粘贴可能存在格式错误,故实验中所需要的YAML文件以及本地包均打包至网盘

链接:https://pan.baidu.com/s/1Zsty68DNZu3gewXVUEabyA 
提取码:1vg4 

1.1  Statefulset控制器:概念、原理解读

StatefulSet是为了管理有状态服务的问题而设计的.

扩展:

什么是有状态服务?

StatefulSet是有状态的集合,管理有状态的服务,它所管理的Pod的名称不能随意变化。数据持久化的目录也是不一样,每一个Pod都有自己独有的数据持久化存储目录。比如MySQL主从、redis集群等。

什么是无状态服务?

RC、Deployment、DaemonSet都是管理无状态的服务,它们所管理的Pod的IP、名字,启停顺序等都是随机的。个体对整体无影响,所有pod都是共用一个数据卷的,部署的tomcat就是无状态的服务,tomcat被删除,在启动一个新的tomcat,加入到集群即可,跟tomcat的名字无关。

1.2  Statefulset资源清单文件编写技巧

#查看定义Statefulset资源需要的字段

[root@xianchaomaster1 ~]# kubectl explain statefulset

KIND:     StatefulSet

VERSION:  apps/v1

DESCRIPTION:

     StatefulSet represents a set of pods with consistent identities. Identities

     are defined as:

     - Network: A single stable DNS and hostname.

     - Storage: As many VolumeClaims as requested. The StatefulSet guarantees

     that a given network identity will always map to the same storage identity.

FIELDS:

   apiVersion     <string> #定义statefulset资源需要使用的api版本

   kind <string>          #定义的资源类型

   metadata <Object>     #元数据

   spec <Object>          #定义容器相关的信息

#查看statefulset.spec字段如何定义?

[root@xianchaomaster1 ~]# kubectl explain statefulset.spec

KIND:     StatefulSet

VERSION:  apps/v1

RESOURCE: spec <Object>

DESCRIPTION:

     Spec defines the desired identities of pods in this set.

     A StatefulSetSpec is the specification of a StatefulSet.

FIELDS:

   podManagementPolicy <string> #pod管理策略

   replicas   <integer>  #副本数

   revisionHistoryLimit     <integer> #保留的历史版本

   selector   <Object> -required- #标签选择器,选择它所关联的pod

   serviceName  <string> -required-  #headless service的名字

   template <Object> -required-     #生成pod的模板

   updateStrategy     <Object>   #更新策略

   volumeClaimTemplates <[]Object> #存储卷申请模板

#查看statefulset的spec.template字段如何定义?

#对于template而言,其内部定义的就是pod,pod模板是一个独立的对象

[root@xianchaomaster1 ~]# kubectl explain statefulset.spec.template

KIND:     StatefulSet

VERSION:  apps/v1

RESOURCE: template <Object>

DESCRIPTION:

     template is the object that describes the pod that will be created if

     insufficient replicas are detected. Each pod stamped out by the StatefulSet

     will fulfill this Template, but have a unique identity from the rest of the

     StatefulSet.

     PodTemplateSpec describes the data a pod should have when created from a

     template

FIELDS:

   metadata <Object>

   spec <Object>  #定义容器属性的

通过上面可以看到,statefulset资源中有两个spec字段。第一个spec声明的是statefulset定义多少个Pod副本(默认将仅部署1个Pod)、匹配Pod标签的选择器、创建pod的模板、存储卷申请模板,第二个spec是spec.template.spec:主要用于Pod里的容器属性等配置。

.spec.template里的内容是声明Pod对象时要定义的各种属性,所以这部分也叫做PodTemplate(Pod模板)。还有一个值得注意的地方是:在.spec.selector中定义的标签选择器必须能够匹配到spec.template.metadata.labels里定义的Pod标签,否则Kubernetes将不允许创建statefulset。

1.3 Statefulset使用案例:部署web站点

#编写一个Statefulset资源清单文件

[root@xianchaomaster1 ~]# cat statefulset.yaml

apiVersion: v1

kind: Service

metadata:

  name: nginx

  labels:

     app: nginx

spec:

  ports:

  - port: 80

    name: web

  clusterIP: None

  selector:

    app: nginx

---

apiVersion: apps/v1

kind: StatefulSet

metadata:

  name: web

spec:

  selector:

    matchLabels:

      app: nginx

  serviceName: "nginx"

  replicas: 2

  template:

    metadata:

     labels:

       app: nginx

    spec:

      containers:

      - name: nginx

        image: nginx

        imagePullPolicy: IfNotPresent

        ports:

        - containerPort: 80

          name: web

        volumeMounts:

        - name: www

          mountPath: /usr/share/nginx/html

  volumeClaimTemplates:

  - metadata:

      name: www

    spec:

      accessModes: ["ReadWriteOnce"]

      storageClassName: "nfs"

      resources:

        requests:

          storage: 1Gi

#更新资源清单文件

[root@xianchaomaster1 ~]# kubectl apply -f statefulset.yaml

service/nginx created

statefulset.apps/web created

      #查看statefulset是否创建成功

[root@xianchaomaster1 ~]# kubectl get statefulset

NAME   READY   AGE

web      2/2      42s

#查看pod

[root@xianchaomaster1 ~]# kubectl get pods -l app=nginx

NAME    READY   STATUS    RESTARTS   AGE

web-0   1/1     Running   0          2m17s

web-1   1/1     Running   0          115s

#通过上面可以看到创建的pod是有序的

#查看headless service

[root@xianchaomaster1 ~]# kubectl get svc -l app=nginx

NAME    TYPE        CLUSTER-IP   EXTERNAL-IP   PORT(S)   AGE

nginx    ClusterIP      None         <none>        80/TCP    3m19s

#查看pvc

[root@xianchaomaster1 ~]# kubectl get pvc

[root@xianchaomaster1 statefulset]# kubectl get pvc

NAME          STATUS   VOLUME                                     CAPACITY   ACCESS MODES   STORAGECLASS   AGE

www-web-0     Bound    pvc-39a9755f-3248-49ff-8f9e-5b068b609c8f   1Gi        RWO,RWX        nfs-web        7m45s

www-web-1     Bound    pvc-be93d4a3-1aca-44cc-802f-ddeb38c05018   1Gi        RWO,RWX        nfs-web        7m41s

#查看pv

[root@xianchaomaster1 ~]# kubectl get pv   

[root@xianchaomaster1 statefulset]# kubectl get pv

NAME                                       CAPACITY   ACCESS MODES   RECLAIM POLICY   STATUS      CLAIM                 STORAGECLASS   REASON   AGE

pvc-39a9755f-3248-49ff-8f9e-5b068b609c8f   1Gi        RWO,RWX        Delete           Bound       default/www-web-0     nfs-web                 8m3s

pvc-be93d4a3-1aca-44cc-802f-ddeb38c05018   1Gi        RWO,RWX        Delete           Bound       default/www-web-1     nfs-web                 7m59s

                   

#查看pod主机名

[root@xianchaomaster1 ~]# for i in 0 1; do kubectl exec web-$i -- sh -c 'hostname';done

web-0

web-1

备注:

StatefulSet由以下几个部分组成:

1. Headless Service:用来定义pod网路标识,生成可解析的DNS记录

2. volumeClaimTemplates:存储卷申请模板,创建pvc,指定pvc名称大小,自动创建pvc,且pvc由存储类供应。

3. StatefulSet:管理pod的

扩展:什么是Headless service?

Headless service不分配clusterIP,headless service可以通过解析service的DNS,返回所有Pod的dns和ip地址 (statefulSet部署的Pod才有DNS),普通的service,只能通过解析service的DNS返回service的ClusterIP。

1.headless service会为service分配一个域名

<service name>.$<namespace name>.svc.cluster.local

K8s中资源的全局FQDN格式:
  Service_NAME.NameSpace_NAME.Domain.LTD.
  Domain.LTD.=svc.cluster.local.     #这是默认k8s集群的域名。

FQDN 全称 Fully Qualified Domain Name

即全限定域名:同时带有主机名和域名的名称

FQDN = Hostname + DomainName

如 主机名是 xianchao

域名是 baidu.com

FQDN= xianchao.baidu.com


2.StatefulSet会为关联的Pod保持一个不变的Pod Name
statefulset中Pod的名字格式为$(StatefulSet name)-$(pod序号)


3.StatefulSet会为关联的Pod分配一个dnsName
$<Pod Name>.$<service name>.$<namespace name>.svc.cluster.local

为什么要用volumeClaimTemplate?

对于有状态应用都会用到持久化存储,比如mysql主从,由于主从数据库的数据是不能存放在一个目录下的,每个mysql节点都需要有自己独立的存储空间。而在deployment中创建的存储卷是一个共享的存储卷,多个pod使用同一个存储卷,它们数据是同步的,而statefulset定义中的每一个pod都不能使用同一个存储卷,这就需要使用volumeClainTemplate,当在使用statefulset创建pod时,volumeClainTemplate会自动生成一个PVC,从而请求绑定一个PV,每一个pod都有自己专用的存储卷。Pod、PVC和PV对应的关系图如下:

#使用kubectl run运行一个提供nslookup命令的容器的,这个命令来自于dnsutils包,通过对pod主机名执行nslookup,可以检查它们在集群内部的DNS地址:

[root@xianchaomaster1 ~]# kubectl run busybox --image docker.io/library/busybox:1.28  --image-pull-policy=IfNotPresent --restart=Never --rm -it busybox -- sh

root@web-1:/# nslookup web-0.nginx.default.svc.cluster.local 

Server:         10.96.0.10

Address:      10.96.0.10#53

Name:  web-0.nginx.default.svc.cluster.local

#statefulset创建的pod也是有dns记录的

Address: 10.244.209.154  #解析的是pod的ip地址

root@web-1:/# nslookup nginx.default.svc.cluster.local

Server:         10.96.0.10

Address:      10.96.0.10#53

Name:  nginx.default.svc.cluster.local  #查询service dns,会把对应的pod ip解析出来

Address: 10.244.209.139

Name:  nginx.default.svc.cluster.local

Address: 10.244.209.140

root@web-1:/# dig -t A nginx.default.svc.cluster.local @10.96.0.10

; <<>> DiG 9.11.5-P4-5.1+deb10u3-Debian <<>> -t A nginx.default.svc.cluster.local @10.96.0.10

;; global options: +cmd

;; Got answer:

;; WARNING: .local is reserved for Multicast DNS

;; You are currently testing what happens when an mDNS query is leaked to DNS

;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 27207

;; flags: qr aa rd; QUERY: 1, ANSWER: 2, AUTHORITY: 0, ADDITIONAL: 1

;; WARNING: recursion requested but not available

;; OPT PSEUDOSECTION:

; EDNS: version: 0, flags:; udp: 4096

; COOKIE: f2f17492cf38ad44 (echoed)

;; QUESTION SECTION:

;nginx.default.svc.cluster.local. IN   A

;; ANSWER SECTION:

nginx.default.svc.cluster.local. 30 IN      A     10.244.209.139

nginx.default.svc.cluster.local. 30 IN      A     10.244.209.140

;; Query time: 0 msec

;; SERVER: 10.96.0.10#53(10.96.0.10)

;; WHEN: Fri Apr 09 15:10:40 UTC 2021

;; MSG SIZE  rcvd: 166

dig的使用

dig -t A nginx.default.svc.cluster.local @10.96.0.10

格式如下: 

@来指定域名服务器

A 为解析类型 ,A记录

-t 指定要解析的类型

A记录:
  A记录是解析域名到IP

资源清单详细解读:

apiVersion: v1  #定义api版本

kind: Service   #定义要创建的资源:service

metadata:

  name: nginx   #定义service的名字

  labels:

     app: nginx  #service的标签

spec:

  ports:

  - port: 80

    name: web

  clusterIP: None  #创建一个没有ip的service

  selector:

    app: nginx     #选择拥有app=nginx标签的pod

---

apiVersion: apps/v1

kind: StatefulSet

metadata:

  name: web

spec:

  selector:

    matchLabels:

      app: nginx

  serviceName: "nginx"  #headless service的名字

  replicas: 2              #副本数

  template:              #定义pod的模板

    metadata:

     labels:

       app: nginx

    spec:

      containers:

      - name: nginx

        image: nginx

        imagePullPolicy: IfNotPresent

        ports:

        - containerPort: 80

          name: web

        volumeMounts:

        - name: www

          mountPath: /usr/share/nginx/html

  volumeClaimTemplates:     #存储卷申请模板

  - metadata:

      name: www

    spec:

      accessModes: ["ReadWriteOnce"]

storageClassName: "nfs"  #指定从哪个存储类申请pv

      resources:

        requests:

          storage: 1Gi   #需要1G的pvc,会自动跟符合条件的pv绑定

Statefulset总结:

1、Statefulset管理的pod,pod名字是有序的,由statefulset的名字-0、1、2这种格式组成

2、创建statefulset资源的时候,必须事先创建好一个service,如果创建的service没有ip,那对这个service做dns解析,会找到它所关联的pod ip,如果创建的service有ip,那对这个service做dns解析,会解析到service本身ip。

3、statefulset管理的pod,删除pod,新创建的pod名字跟删除的pod名字是一样的

4、statefulset具有volumeclaimtemplate这个字段,这个是卷申请模板,会自动创建pv,pvc也会自动生成,跟pv进行绑定,那如果创建的statefulset使用了volumeclaimtemplate这个字段,那创建pod,数据目录是独享的

5、ststefulset创建的pod,是域名的(域名组成:pod-name.svc-name.svc-namespace.svc.cluster.local)

扩展:

举例说明service 和headless service区别:

1、通过deployment创建pod,pod前端创建一个service

[root@xianchaomaster1 ~]# cat deploy-service.yaml

apiVersion: v1

kind: Service

metadata:

  name: my-nginx

  labels:

    run: my-nginx

spec:

  type: ClusterIP

  ports:

  - port: 80   #service的端口,暴露给k8s集群内部服务访问

    protocol: TCP

    targetPort: 80    #pod容器中定义的端口

  selector:

    run: my-nginx  #选择拥有run=my-nginx标签的pod

---

apiVersion: apps/v1

kind: Deployment

metadata:

  name: my-nginx

spec:

  selector:

    matchLabels:

      run: my-nginx

  replicas: 2

  template:

    metadata:

      labels:

        run: my-nginx

    spec:

      containers:

      - name: my-nginx

        image: busybox

        imagePullPolicy: IfNotPresent

        ports:

        - containerPort: 80

        command:

          - sleep

          - "3600"

#更新资源清单文件

[root@xianchaomaster1 ~]# kubectl apply -f deploy-service.yaml

#查看service

[root@xianchaomaster1 ~]# kubectl get svc -l run=my-nginx

NAME       TYPE        CLUSTER-IP     EXTERNAL-IP   PORT(S)  

my-nginx   ClusterIP     10.100.89.90   <none>        80/TCP   

#查看pod

[root@xianchaomaster1 ~]# kubectl get pods -l run=my-nginx

NAME                        READY   STATUS    RESTARTS   AGE

my-nginx-58f74fc5b6-jzbvk   1/1     Running   0          70s

my-nginx-58f74fc5b6-n9lqv   1/1     Running   0          53s 

#通过上面可以看到deployment创建的pod是随机生成的

#进入到web-1的pod

[root@xianchaomaster1 ~]# kubectl exec -it web-1 -- /bin/bash

root@web-1:/# nslookup my-nginx.default.svc.cluster.local

Server:         10.96.0.10

Address:      10.96.0.10#53

Name:  my-nginx.default.svc.cluster.local

Address: 10.100.89.90   #解析的是service的ip地址

Statefulset总结:

1、Statefulset管理的pod,pod名字是有序的,由statefulset的名字-0、1、2这种格式组成

2、创建statefulset资源的时候,必须事先创建好一个service,如果创建的service没有ip,那对这个service做dns解析,会找到它所关联的pod ip,如果创建的service有ip,那对这个service做dns解析,会解析到service本身ip。

3、statefulset管理的pod,删除pod,新创建的pod名字跟删除的pod名字是一样的

4、statefulset具有volumeclaimtemplate这个字段,这个是卷申请模板,会自动创建pv,pvc也会自动生成,跟pv进行绑定,那如果创建的statefulset使用了volumeclaimtemplate这个字段,那创建pod,数据目录是独享的

5、ststefulset创建的pod,是域名的(域名组成:pod-name.svc-name.svc-namespace.svc.cluster.local)

1.4 Statefulset管理pod:扩容、缩容、更新

#Statefulset实现pod的动态扩容

如果我们觉得两个副本太少了,想要增加,只需要修改配置文件statefulset.yaml里的replicas的值即可,原来replicas: 2,现在变成replicaset: 3,修改之后,执行如下命令更新:

[root@xianchaomaster1 ~]# kubectl apply -f statefulset.yaml

service/nginx unchanged

statefulset.apps/web configured

[root@xianchaomaster1 ~]# kubectl get sts

NAME   READY   AGE

web    3/3     60m

      [root@xianchaomaster1 ~]# kubectl get pods -l app=nginx

NAME    READY   STATUS    RESTARTS   AGE

web-0   1/1     Running   0          61m

web-1   1/1     Running   0          60m

web-2   1/1     Running   0          79s

      #也可以直接编辑控制器实现扩容

[root@xianchaomaster1 ~]# kubectl edit sts web

#这个是我们把请求提交给了apiserver,实时修改

把上面的spec下的replicas 后面的值改成4,保存退出

[root@xianchaomaster1 ~]#  kubectl get pods -l app=nginx

NAME    READY   STATUS    RESTARTS   AGE

web-0   1/1     Running   0          62m

web-1   1/1     Running   0          62m

web-2   1/1     Running   0          3m13s

web-3   1/1     Running   0          26s

#Statefulset实现pod的动态缩容

如果我们觉得4个Pod副本太多了,想要减少,只需要修改配置文件statefulset.yaml里的replicas的值即可,把replicaset:4变成replicas: 2,修改之后,执行如下命令更新:

[root@xianchaomaster1 ~]# kubectl apply -f statefulset.yaml

service/nginx unchanged

statefulset.apps/web configured

[root@xianchaomaster1 ~]#  kubectl get pods -l app=nginx

NAME    READY   STATUS    RESTARTS   AGE

web-0   1/1     Running   0          64m

web-1   1/1     Running   0          64m

#Statefulset实现pod的更新

[root@xianchaomaster1 ~]# kubectl explain sts.spec.updateStrategy

[root@xianchaomaster1]# vim statefulset.yaml

在一个终端动态查看pod

[root@xianchaomaster1 ~]# kubectl get pods -l app=nginx -w

另一个终端执行如下命令:

[root@xianchaomaster1 ]# kubectl apply -f statefulset.yaml

在一个终端动态查看pod

[root@xianchaomaster1 ~]# kubectl get pods -l app=nginx -w

出现的结果如下:

web-0                              1/1     Running   0          10m

web-1                              1/1     Running   0          10m

web-1                              1/1     Terminating   0          10m

web-1                              1/1     Terminating   0          10m

web-1                              0/1     Terminating   0          10m

web-1                              0/1     Terminating   0          10m

web-1                              0/1     Terminating   0          10m

web-1                              0/1     Pending       0          0s

web-1                              0/1     Pending       0          0s

web-1                              0/1     ContainerCreating   0          0s

web-1                              0/1     ContainerCreating   0          1s

web-1                              1/1     Running             0          2s

web-1                              1/1     Running             0          11s

从上面结果可以看出来,pod在更新的时候,只是更新了web-1这个pod, partition: 1表示更新的时候会把pod序号大于等于1的进行更新

如果更新策略是OnDelete,那不会自动更新pod,需要手动删除,重新常见的pod才会实现更新

Logo

K8S/Kubernetes社区为您提供最前沿的新闻资讯和知识内容

更多推荐