k8s StorageClass(NFS)

本文介绍了如何在Kubernetes环境中利用NFS动态创建StorageClass和PersistentVolume。首先,详细解释了StorageClass的作用及其重要属性,如Provisioner和ReclaimPolicy。接着,演示了部署nfs-client的步骤,包括创建ServiceAccount、ClusterRole、ClusterRoleBinding以及Deployment。然后,创建了nfs-client的StorageClass,并展示了如何创建PersistentVolumeClaim。最后,通过创建一个Pod测试了动态分配的存储空间,证实了PVC成功绑定到PV并使用NFS存储。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一. StorageClass 简介

在K8S环境,当pod需要存储空间时,StorageClass比PV更灵活和方便,每个 StorageClass 都包含 provisioner、parameters 和 reclaimPolicy 字段, 这些字段会在 StorageClass 需要动态分配 PersistentVolume 时会使用到。

StorageClass的属性
• Provisioner(存储分配器):用来决定使用哪个卷插件分配 PV,该字段必须指定。可以指 定内部分配器,也可以指定外部分配器。外部分配器的代码地址为: kubernetesincubator/external-storage,其中包括NFS和Ceph等。
• Reclaim Policy(回收策略):通过reclaimPolicy字段指定创建的Persistent Volume的回收 策略,回收策略包括:Delete 或者 Retain,没有指定默认为Delete。

二. 部署nfs-client

这里我们以NFS为例,要使用NFS,我们就需要一个nfs-client的自动装载程序,我们称之为Provisioner,这个程序会使用我们已经配置好的NFS服务器自动创建持久卷,也就是自动帮我们创建PV。
详细可以参考:https://github.com/kubernetes-retired/external-storage/blob/master/nfs-client/deploy/deployment.yaml

在部署之前,首先得确保有可用得NFS服务器,这里默认已经有可用得NFS服务器了。

  1. 创建ServiceAccount,为nfs-client授权。nfs-client-sa.yaml
---
apiVersion: v1
kind: ServiceAccount
metadata:
  name: nfs-client-provisioner

---
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
  name: nfs-client-provisioner-clusterrole
rules:
  - apiGroups: [""]
    resources: ["persistentvolumes"]
    verbs: ["get", "list", "watch", "create", "delete"]
  - apiGroups: [""]
    resources: ["persistentvolumeclaims"]
    verbs: ["get", "list", "watch", "update"]
  - apiGroups: ["storage.k8s.io"]
    resources: ["storageclasses"]
    verbs: ["get", "list", "watch"]
  - apiGroups: [""]
    resources: ["events"]
    verbs: ["list", "watch", "create", "update", "patch"]
  - apiGroups: [""]
    resources: ["endpoints"]
    verbs: ["create", "delete", "get", "list", "watch", "patch", "update"]

---
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRoleBinding
metadata:
  name: nfs-client-provisioner-clusterrolebinding
subjects:
- kind: ServiceAccount
  name: nfs-client-provisioner
  namespace: default
roleRef:
  kind: ClusterRole
  name: nfs-client-provisioner-clusterrole
  apiGroup: rbac.authorization.k8s.io

通过上面得配置,设置nfs-client对PV,PVC,StorageClass等规则。接下来我们创建这个YAML文件:

[root@baozexu storageclass]# kubectl apply -f nfs-client-sa.yaml 
serviceaccount/nfs-client-provisioner created
clusterrole.rbac.authorization.k8s.io/nfs-client-provisioner-clusterrole created
clusterrolebinding.rbac.authorization.k8s.io/nfs-client-provisioner-clusterrolebinding created
  1. 创建nfs-client

    使用Deployment来创建nfs-client,配置如下:nfs-client.yaml

    apiVersion: apps/v1
    kind: Deployment
    metadata:
      name: nfs-client-provisioner
      labels:
        app: nfs-client-provisioner
      # replace with namespace where provisioner is deployed
      namespace: default
    spec:
      replicas: 1
      strategy:
        type: Recreate
      selector:
        matchLabels:
          app: nfs-client-provisioner
      template:
        metadata:
          labels:
            app: nfs-client-provisioner
        spec:
          serviceAccountName: nfs-client-provisioner
          containers:
            - name: nfs-client-provisioner
              image: easzlab/nfs-subdir-external-provisioner:v4.0.1
              volumeMounts:
                - name: nfs-client-root
                  mountPath: /persistentvolumes
              env:
                - name: PROVISIONER_NAME
                  value: rookieops/nfs
                - name: NFS_SERVER
                  value: 2.16.128.1 
                - name: NFS_PATH
                  value: /public 
          volumes:
            - name: nfs-client-root
              nfs:
                server: 2.16.128.1
                path: /public

    然后创建这个yaml文件

    [root@baozexu storageclass]# kubectl apply -f nfs-client.yaml 
    deployment.extensions/nfs-client-prosioner created

    查看其状态

    [root@baozexu nfs]# kubectl get pods
    NAME                                    READY   STATUS             RESTARTS   AGE
    nfs-client-provisioner-6c59bb8b-k8hd8   1/1     Running            0          50m
    1. 上面创建完成后就可以创建StorageClass了

​ nfs-client-storageclass.yaml

apiVersion: storage.k8s.io/v1
kind: StorageClass
metadata:
  name: nfs-client-storageclass
provisioner: rookieops/nfs

​ 注意provisioner必须和上面得Deployment的YAML文件中PROVISIONER_NAME的值保持一致。

​ 创建这个yaml文件:

[root@baozexu nfs]# kubectl apply -f nfs-client-storageclass.yaml 
storageclass.storage.k8s.io/nfs-client-storageclass created
[root@baozexu nfs]# kubectl get storageclasses.storage.k8s.io 
NAME                      PROVISIONER             RECLAIMPOLICY   VOLUMEBINDINGMODE      ALLOWVOLUMEEXPANSION   AGE
nfs-client-storageclass   rookieops/nfs           Delete          Immediate              false                  4h30m

注意:卷绑定模式

volumeBindingMode 字段控制了卷绑定和动态制备 应该发生在什么时候。

默认情况下,Immediate 模式表示一旦创建了 PersistentVolumeClaim 也就完成了卷绑定和动态制备。 对于由于拓扑限制而非集群所有节点可达的存储后端,PersistentVolume 会在不知道 Pod 调度要求的情况下绑定或者制备。

集群管理员可以通过指定 WaitForFirstConsumer 模式来解决此问题。 该模式将延迟 PersistentVolume 的绑定和制备,直到使用该 PersistentVolumeClaim 的 Pod 被创建。 PersistentVolume 会根据 Pod 调度约束指定的拓扑来选择或制备。这些包括但不限于 资源需求节点筛选器pod 亲和性和互斥性、 以及污点和容忍度

  1. 创建pvc
kind: PersistentVolumeClaim
apiVersion: v1
metadata:
  name: pvc1
spec:
  storageClassName: nfs-client-storageclass
  accessModes:
    - ReadWriteMany
  resources:
    requests:
      storage: 1Gi
  1. 创建这个yaml文件,观察其状态:
[root@nfs baozexu]# kubectl apply -f test-pvc.yaml 
persistentvolumeclaim/test-nfs-pvc created
[root@baozexu nfs]# kubectl get pvc
NAME   STATUS   VOLUME                                     CAPACITY   ACCESS MODES   STORAGECLASS              AGE
pvc1   Bound    pvc-33d2aa7f-e52f-4865-a05e-ebd53fbec914   1Gi        RWX            nfs-client-storageclass   120m

​ 我们看到该PVC自动申请到空间,其STORAGECLASS就是我们创建的nfs-client-storageclass。

​ 6. 创建一个pod,进行测试 pv-pod.yaml

apiVersion: v1
kind: Pod
metadata:
  name: storageclass-pod
spec:
  containers:
  - name: busybox
    image: busybox:latest
    imagePullPolicy: IfNotPresent
    command:
    - "/bin/sh"
    - "-c"
    args:
    - "sleep 3600"
    volumeMounts:
    - name: nfs-pvc
      mountPath: /mnt
  restartPolicy: Never
  volumes:
  - name: nfs-pvc
    persistentVolumeClaim:
      claimName: pvc1
  1. 查看pv是否生成
[root@baozexu ~]# kubectl get pv
NAME                                       CAPACITY   ACCESS MODES   RECLAIM POLICY   STATUS   CLAIM                                 STORAGECLASS              REASON   AGE
pvc-33d2aa7f-e52f-4865-a05e-ebd53fbec914   1Gi        RWX            Delete           Bound    default/pvc1                          nfs-client-storageclass            77m
  1. 然后查看nfs目录中是否有文件生成
[root@baozexu public]# ls
default-pvc1-pvc-33d2aa7f-e52f-4865-a05e-ebd53fbec914

我们可以看到生成了对应的目录,格式和我们上面说的一致。

### 如何在Ubuntu中设置Kubernetes NFS StorageClass配置示例及故障排除 #### 安装NFS服务器 为了使NFS能够正常工作,在Ubuntu上需更新软件列表并安装`nfs-kernel-server`,这可以通过执行命令来完成[^1]。 ```bash sudo apt update && sudo apt install -y nfs-kernel-server ``` #### 创建共享目录与导出配置 创建一个用于挂载的文件夹,并编辑`/etc/exports`文件以允许特定网络内的客户端访问此路径。例如: ```bash sudo mkdir -p /srv/nfs/kube-pv echo "/srv/nfs/kube-pv *(rw,sync,no_subtree_check)" | sudo tee -a /etc/exports ``` 重启NFS服务以便应用更改: ```bash sudo systemctl restart nfs-kernel-server ``` #### 配置StorageClass资源对象 定义YAML格式描述符来声明持久化卷类(PVC),其中指定了provisioner参数为外部插件名称或内部驱动程序标识符。对于简单的静态PV分配方案,则不需要动态供应器;而对于自动化的PVC绑定过程则必需指定合适的provisioner实现方式。 下面是一个适用于大多数场景下的通用模板: ```yaml apiVersion: storage.k8s.io/v1 kind: StorageClass metadata: name: managed-nfs-storage provisioner: example.com/nfs # 或者使用社区支持版本如 "fuseim.pri/ifs" parameters: archiveOnDelete: "false" reclaimPolicy: Retain # 可选值有 Delete 和 Retain,默认是 Delete mountOptions: - hard - nolock allowVolumeExpansion: true # 启用扩展功能 volumeBindingMode: WaitForFirstConsumer ``` 注意替换上述代码中的`example.com/nfs`部分为实际使用的第三方组件提供的有效字符串表示形式之一。 #### 故障排查指南 当遇到问题时可以按照如下方法进行初步诊断: - 使用kubectl describe查看Pod状态日志; - 查看Node节点上的/var/log/syslog是否有任何关于NFS错误的信息记录; - 确认防火墙规则是否阻止了必要的端口通信(默认情况下TCP 2049); - 测试从任意集群成员机器到NFS Server主机之间的连通性和性能表现。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值