在 Kubernetes 中遇见 NFS 存储
当在 K8s 中使用到 NFS 类型的存储时,首先需要搭建一个 NFS 服务,另外注意一个挂载路径问题,就能愉快地玩耍它啦。
安装NFS服务
1 | sudo apt install -y nfs-kernel-server rpcbind |
创建目录 ~/nfs
作为共享目录,并在该目录下,新建一个文件 date > time.txt
。
配置NFS服务
添加共享目录配置:/etc/exports
1 | # /etc/exports: the access control list for filesystems which may be exported |
访问权限选项
- rw 可读写
- ro 只读
用户映射选项
- all_squash 将远程访问的所有普通用户及所属组都映射为匿名用户或用户组(nfsnobody)
- no_all_squash:与all_squash取反(默认设置)
- root_squash:将root用户及所属组都映射为匿名用户或用户组(默认设置)
- no_root_squash:与rootsquash取反
- anonuid=xxx:将远程访问的所有用户都映射为匿名用户,并指定该用户为本地用户(UID=xxx)
- anongid=xxx:将远程访问的所有用户组都映射为匿名用户组账户,并指定该匿名用户组账户为本地用户组账户(GID=xxx)
其它选项
- secure:限制客户端只能从小于1024的tcp/ip端口连接nfs服务器(默认设置)
- insecure:允许客户端从大于1024的tcp/ip端口连接服务器
- sync:将数据同步写入内存缓冲区与磁盘中,效率低,但可以保证数据的一致性
- async:将数据先保存在内存缓冲区中,必要时才写入磁盘
- wdelay:检查是否有相关的写操作,如果有则将这些写操作一起执行,这样可以提高效率(默认设置)
- no_wdelay:若有写操作则立即执行,应与sync配合使用
- subtree:若输出目录是一个子目录,则nfs服务器将检查其父目录的权限(默认设置)
- no_subtree:即使输出目录是一个子目录,nfs服务器也不检查其父目录的权限,这样可以提高效率
其他操作
启动/重启NFS服务:/etc/init.d/nfs-kernel-server
1 | sudo /etc/init.d/nfs-kernel-server start |
刷新NFS配置:exportfs
1 | # 重新读取 /etc/exports 配置信息 |
查看共享的目录:showmount
1 | # 本机共享目录 |
在别的机器上面进行 mount 操作
1 | sudo mount -t nfs <nfs-server-ip>:/home/ubuntu/nfs /data/backups -o nolock |
进入该目录,可以看到之前的 time.txt
文件,进行修改后,可在服务器所在机器上看到变更。
1 | $ cd /data/backups |
卸载目录
1 | sudo umount /data/backups |
权限问题说明
- 客户端连接时候,对普通用户的检查
- 如果明确设定了普通用户被映射的身份,那么此时客户端用户的身份转换为指定用户;
- 如果NFS server上面有同名用户,那么此时客户端登录账户的身份转换为NFS server上面的同名用户;
- 如果没有明确指定,也没有同名用户,那么此时用户身份被映射成nfsnobody;
- 客户端连接的时候,对root的检查
- 如果设置no_root_squash,那么此时root用户的身份被映射为NFS server上面的root;
- 如果设置了all_squash、anonuid、anongid,此时root 身份被映射为指定用户;
- 如果没有明确指定,此时root用户被映射为nfsnobody;
- 如果同时指定no_root_squash与all_squash 用户将被映射为 nfsnobody,如果设置了anonuid、anongid将被映射到所指定的用户与组;
Kubernetes中添加NFS支持
- csi-driver-nfs/charts at master · kubernetes-csi/csi-driver-nfs (github.com)
- csi-driver-nfs/driver-parameters.md at master · kubernetes-csi/csi-driver-nfs (github.com)
添加 StorageClass
1 | apiVersion: storage.k8s.io/v1 |
遇到的坑
如果 NFS 服务 /etc/exports
里设置的共享路径是 /home/ubuntu/nfs
,在 StorageClass
中 share
字段如果填 /
的话,挂载时会报权限不足。
通过 csi-driver-nfs 的日志可以发现,在创建 PV 时,用的路径是 /pod-xxxxxx
。
Pod 启动失败,报
0/1 nodes are available: 1 pod has unbound immediate PersistentVolumeClaims
。PVC 处于 Pending 状态,报
failed to provision volume with StorageClass "nfs-csi": rpc error: code = Internal desc = failed to make subdirectory: mkdir /tmp/pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2/pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2: read-only file system
。csi-nfs-controller
日志:1
2
3
4
5
6
7
8
9
10
11I0720 09:53:52.378454 1 controllerserver.go:285] internally mounting 192.168.59.102:/ at /tmp/pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2
I0720 09:53:52.378935 1 controllerserver.go:300] internally unmounting /tmp/pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2
I0720 09:53:52.378942 1 nodeserver.go:163] NodeUnpublishVolume: unmounting volume 192.168.59.102##pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2# on /tmp/pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2
I0720 09:53:52.378955 1 mount_helper_common.go:99] "/tmp/pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2" is a mountpoint, unmounting
I0720 09:53:52.378959 1 mount_linux.go:294] Unmounting /tmp/pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2
W0720 09:53:52.387057 1 mount_helper_common.go:133] Warning: "/tmp/pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2" is not a mountpoint, deleting
I0720 09:53:52.387102 1 nodeserver.go:168] NodeUnpublishVolume: unmount volume 192.168.59.102##pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2# on /tmp/pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2 successfully
E0720 09:53:52.387108 1 utils.go:93] GRPC error: rpc error: code = Internal desc = failed to make subdirectory: mkdir /tmp/pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2/pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2: read-only file system
I0720 09:54:48.201408 1 utils.go:88] GRPC call: /csi.v1.Controller/CreateVolume
I0720 09:54:48.201568 1 utils.go:89] GRPC request: {"capacity_range":{"required_bytes":1000000000},"name":"pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2","parameters":{"csi.storage.k8s.io/pv/name":"pvc-ca0ab165-9aa5-42a4-ae15-f4fd036142d2","csi.storage.k8s.io/pvc/name":"data-primary-mariadb-cluster-primary-0","csi.storage.k8s.io/pvc/namespace":"default","mountPermissions":"0777","server":"192.168.59.102","share":"/"},"volume_capabilities":[{"AccessType":{"Mount":{}},"access_mode":{"mode":7}}]}在其他 Linux 机器上面,使用 mount 命令尝试挂载:
1
2
3
4
5
6
7
8
9$ sudo mount -t nfs 192.168.59.102:/ /tmp/nfs
$ ls /tmp/nfs/
home
$ ls /tmp/nfs/home
ubuntu
$ ls /tmp/nfs/home/ubuntu
nfs
$ ls /tmp/nfs/home/ubuntu/nfs
pvc-1e8ff556-d183-4471-92cd-3869ffbcb589 pvc-9f429e24-c6da-45c9-9860-f4e338f61068 time.txt
看到这里本能地想到了
StorageClass
share 字段应当修改成/home/ubuntu/nfs
如果填写 /
,会将 NFS 服务上的 /
挂载在 K8s 所在机器的 /tmp/pvc-xxxx
下,当在 /tmp/pvc-xxx
目录中创建子目录时,即在 NFS 服务器上的 /
下创建目录,理所应当会失败。
Reference
在 Kubernetes 中遇见 NFS 存储