docker(4): 持久化

一. 概述

容器中的数据可以存储在容器层。但是将数据存放在容器层存在以下问题:

  • 数据不是持久化。意思是如果容器删除了,这些数据也就没了
  • 主机上的其它进程不方便访问这些数据
  • 对这些数据的I/O会经过存储驱动,然后到达主机,引入了一层间接层,因此性能会有所下降

二. Docker 提供了3种持久化数据的方式:

1. bind mount:将宿主机中的文件、目录mount到容器上。其上的数据可以被宿主机读写,可以被mount它的所有容器读写。

2. volume:volume由docker管理,比如创建、删除什么的。默认情况下,volume的存储空间来自于宿主机文件系统中的某个目录,如/var/lib/docker/volumes/,docker系统这外的程序不应该修改其中的数据。volume是官方推荐的持久化方案。

3. tmpfs mount:tmpfs类型文件与普通文件的区别是只存在于宿主机内存中,不会持久化。

2.1 volumes

如果没有显式创建,一个卷会在最开始挂载时被创建。当容器停止时,卷仍然存在。多个容器可以通过read-write或read-only的方式使用同一个卷。

只有在显式删除时,卷才会被删除。如果将一个空卷挂载到容器中一个存有文件或目录的目录中,这些文件或目录会被拷贝到空卷中;如果将一个非空卷挂载到容器中一个存有文件或目录的目录中,这些文件或目录会被隐藏。

使用

  • 创建:docker volume create
  • 删除某个卷:docker volume rm 卷名
  • 删除所有未使用的卷:docker volume prune
  • 列出所有卷:docker volume ls
  • 查看某个卷的信息:docker volume inspect 卷名
  • 挂载到容器:-v或--volume。如果是Docker17.06或更高:推荐使用--mount。(同 bind mount)
    • 挂载类型:key为type,value为bind、volume或tmpfs
    • 挂载源:key为source或src,对于命名卷,value为卷名,对于匿名卷,则忽略
    • 容器中的挂载点:key为destination、dst或target,value为容器中的路径
    • 读写类型:value为readonly,没有key
    • volume-opt选项,可以出现多次。比如volume-driver=local,volume-opt=type=nfs,…
    • 第一个域:对于命名卷,为卷名;匿名卷,则忽略,此时会创建匿名卷
    • 第二个域:容器中的挂载点
    • 第三个域:可选参数,由','隔开,如ro
    • -v或—volume:由3个域组成,’:’分隔
    • —mount:由多个’,’隔开的键值对 = 组成:

当使用docker service create 启动Docker服务时,只支持--mount,不支持-v和--volume。并且每个服务容器使用它们各自的本地卷,因此如果使用本地(local)卷驱动,容器无法通过卷共享数据,但是一些卷驱动支持共享存储。Docker for AWS和Doocker for Azure都使用Cloundstor plugin支持持久存储

场景

  • 多个运行容器间共享数据
  • 当Docker主机不确保具有给定的目录或文件时。卷可以将容器运行时与Docker主机的配置解耦合
  • 备份、恢复、或将数据从一个Docker主机迁移到另一个Docker主机时

2.2 bind mount

主机中的文件或目录通过全路径被引用。在使用绑定挂载时,这些目录或文件不一定要已经存在。

如果使用这种方式将一个目录挂载到容器中一个存有文件或目录的目录中,这些文件或目录会被隐藏;如果主机中的文件或目录不存在,当使用--mount挂载时,Docker会报错,当使用-v或--volume时,会在主机上创建目录

使用

挂载到容器:-v或—volume。如果是Docker17.06或更高:推荐使用—mount。(同 volumes)

  • -v或--volume:由3个域组成,':'分隔
    • 第一个域:对于命名卷,为卷名;匿名卷,则忽略,此时会创建匿名卷
    • 第二个域:容器中的挂载点
    • 第三个域:可选参数,由','隔开,如ro
  • --mount:由多个','隔开的键值对<key>=<value>组成:
    • 挂载类型:key为type,value为bind、volume或tmpfs
    • 挂载源:key为source或src,value为主机中文件或目录的路径
    • 容器中的挂载点:key为destination、dst或target,value为容器中的路径
    • 读写类型:value为readonly,没有key
    • bind-propagation选项:key为bind-propagation,value为rprivate、private、rshared、shared、rslave或slave
    • 一致性选项:value为consistent、delegated、cached。这个选项仅仅适用于Docker for Mac
    • --mount不支持z和Z(这个不同于-v和—volume)

场景

大体上来说,只要可能,最好使用volumes

  • 主机与容器共享配置文件(Docker默认情况下通过这种方式为容器提供DNS解析,通过将/etc/resolv.conf挂载到容器中)
  • 共享源代码或build artifacts(比如将Maven的target/目录挂载到容器中,每次在Docker主机中build Maven工程时,容器能够访问到那些rebuilt artifacts)
  • 当 docker主机中的文件或目录结构和容器需要的一致时

2.3 bind propagation

对于bind mount和volumes,默认都是rprivate。只有在使用bind mount时可配置,且必须在linux下。bind propagation是个超前主题,对于大多数用户来说,并不需要配置

对于一个挂载点/mnt,假设它同时也被挂载到/tmp。bind propagation控制 whether a mount on /tmp/a would also be available on /mnt/a

在设置bind propagation之前,主机文件系统需要支持bind propagation

下面的例子将主机中的target/挂载到容器中2次:

docker run -d

-it

--name devtest

--mount type=bind,source="$(pwd)"/target,target=/app

--mount type=bind,source="$(pwd)"/target,target=/app2,readonly,bind-propagation=rslave

nginx:latest

此时如果创建/app/foo/,/app2/foo也会存在

selinux label

你能添加z或Z选项来修改挂载到容器中的主机文件或目录的selinux label:

  • z选项指明bind mount的内容在多个容器间是共享的
  • Z选项指明bind mount的内容是私有不共享的

要特别小心的使用这两个选项。”Bind-mounting a system directory such as /homeor /usrwith the Zoption renders your host machine inoperable and you may need to relabel the host machine files by hand”

tmpfs mount

只在linux中支持

相对于volumes和bind mount,tmpfs mount是临时的,只在主机内存中持久化。当容器停止,tmpfs mount会被移除。对于临时存放敏感文件很有用

不同于volumes和bind mount,多个容器无法共享tmpfs mount

使用

  • 挂载到容器:—tmpfs。如果是Docker17.06或更高:推荐使用—mount
    • 挂载类型:key为type,value为bind、volume或tmpfs
    • 容器中的挂载点:key为destination、dst或target,value为容器中的路径
    • tmpfs-size和tmpfs-mode选项
    • —tmpfs:直接指定容器中的挂载点。不允许指定任何配置选项
    • —mount:由多个’,’隔开的键值对 = 组成:

场景

  • 最好的使用场景是你既不想将数据存于主机,又不想存于容器中时。这可以是出于安全的考虑,或当应用需要写大量非持久性的状态数据时为了保护容器的性能

volume drivers

机器间共享数据

当构建错误容忍应用时,可能需要配置同一个服务的多个副本来访问相同的文件:

有多种方法来实现这个目的:

  • 为应用添加逻辑,将文件存储到一个云对象存储系统(如Amazon S3)中
  • 使用一个支持将文件写入外部存储系统(如NFS或Amazon S3)的driver来创建卷

volume drivers可以将底层存储系统从应用逻辑中抽象出来。比如,如果你的服务使用一个具有NFS driver的卷,你能更新你的服务使用不同的driver,作为在云中存储数据的示例,而不更改应用程序逻辑

使用

在使用docker volume create或驱动容器创建匿名卷时,可以指定一个volume drivers。下面的例子使用vieux/sshfs作为volume drivers

假设有2个节点,第一个节点是Docker主机,它能SSH到第二个节点

1、在Docker主机中,安装vieux/sshfs插件

docker plugin install --grant-all-permissions vieux/sshfs

2、使用卷驱动创建卷

1)创建命名卷

docker volume create --driver vieux/sshfs

-o sshcmd=test@node2:/home/test

-o password=testpassword

sshvolume

2)启动容器时使用卷驱动创建匿名卷

docker run -d

--name sshfs-container

--volume-driver vieux/sshfs

--mount src=sshvolume,target=/app,volume-opt=sshcmd=test@node2:/home/test,volume-opt=password=testpassword

nginx:latest

3、备份、恢复、迁移数据卷

1)备份一个容器

docker run --rm --volumes-from dbstore -v $(pwd):/backup ubuntu tar cvf /backup/backup.tar /dbdata

  • 启动一个新容器,挂载dbstore容器中的卷
  • 挂载一个本地主机目录到容器/backup
  • 使用tar将dbdata卷中的数据打包成backup.tar

2)用备份恢复容器

使用刚刚创建的备份来恢复容器:

docker run -v /dbdata --name dbstore2 ubuntu /bin/bash

然后,在新创建的容器的卷中使用tar解包备份的数据:

docker run --rm --volumes-from dbstore2 -v $(pwd):/backup ubuntu bash -c "cd /dbdata && tar xvf /backup/backup.tar --strip 1"

参考:

Docker 数据持久化的三种方案,你总能用到_容器

Docker容器数据持久化_五星上炕的博客-CSDN博客_docker数据持久化

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值