postgresql - Docker volume 不持久化数据

标签 postgresql docker dockerfile

这是我的 docker 文件:

FROM ubuntu:14.04

RUN apt-key adv --keyserver keyserver.ubuntu.com --recv-keys B97B0AFCAA1A47F044F244A07FCC7D46ACCC4CF8
RUN echo "deb http://apt.postgresql.org/pub/repos/apt/ precise-pgdg main" > /etc/apt/sources.list.d/pgdg.list

RUN apt-get update && apt-get -y -q install python-software-properties software-properties-common \
    && apt-get -y -q install postgresql-9.3 postgresql-client-9.3 postgresql-contrib-9.3

USER postgres
RUN /etc/init.d/postgresql start \
    && psql --command "CREATE USER pguser WITH SUPERUSER PASSWORD 'pguser';" \
    && createdb -O pguser pgdb
USER root
RUN echo "host all  all    0.0.0.0/0  md5" >> /etc/postgresql/9.3/main/pg_hba.conf
RUN echo "listen_addresses='*'" >> /etc/postgresql/9.3/main/postgresql.conf
EXPOSE 5432

RUN mkdir -p /var/run/postgresql && chown -R postgres /var/run/postgresql
VOLUME  ["/etc/postgresql", "/var/log/postgresql", "/var/lib/postgresql"]
USER postgres

CMD ["/usr/lib/postgresql/9.3/bin/postgres", "-D", "/var/lib/postgresql/9.3/main", "-c", "config_file=/etc/postgresql/9.3/main/postgresql.conf"]

这是我做的...

  1. 我构建 docker 镜像:

    docker build --rm=true -t my_image/postgresql:9.3
    
  2. 然后,我在当前目录中创建一个名为 data 的新目录并运行以下命令:

    docker run -i -t -v="data:/data" -p 5432:5432 my_image/postgresql:9.3
    
  3. 我打开另一个终端并通过运行进入 postgres shell:

    psql -h my_docker_ip -p 5432 -U pguser -W pgdb
    

    然后我创建了一个表:

    pgdb=# create table test (test_id bigserial primary key);
    
  4. 我使用 \dt 验证表是否存在并退出 postgres shell

  5. 我终止 docker 进程并重新运行以下命令:

    docker run -i -t -v="data:/data" -p 5432:5432 my_image/postgresql:9.3
    
  6. 我再次进入 posgrest shell 并运行 \dt

  7. 我注意到
    • 没有 table 。
    • data 目录中没有文件。

我一定是做错了什么,因为我假设我创建的表会持续存在。有人可以指出我的错误吗?

最佳答案

有些事情让我很困惑,官方文档中对我来说不是很清楚。

据我所知,可以通过三种方式创建持久卷。

  • 在容器调用时间,包括完整路径(-v ~/database:/data):使来自主机的外部文件夹在 docker 容器内可用。两者都可以修改它。
  • 在容器调用时间使用卷名称(-v datamysql:/data):使容器内的卷持久可用。如果它不存在,则创建它。您可以使用 docker volume ls 按名称列出它们。在内部,它将存储在一个地方,例如 /var/lib/docker/volumes/ae4445f7c9317a22fe84726fb894c47754f38a7fd150c00fd877024889968750/_data
  • 在容器构建时(Dockerfile 中的 VOLUME ["/database/data"])。每次调用 docker run 都会创建一个新卷,即使您删除了容器,该卷也会保留。这可能会造成混淆,因为后续调用将导致创建不同 卷,这些卷将不会被重复使用。

您可以列出命名(第二种情况)和未命名(第三种情况)卷

$ docker volume ls                                                             
DRIVER              VOLUME NAME                                                             
local               064593b3e65977097d4d0c8402a6c633f1af69be2937bf118678ab8f97ee9a7e               
local               4753ad0437d13e54c76d9c34a30a1843396a1866a0cf9237d500fdcca0d78c5f           
local               8d7a35354f666b2e8a26866a35bbae36bb9601701d4c6b505ab8ce6629f69415               
local               db48eefe8f189b36107ca9c4eebb792690590ab0ba055e7e4e2c9adfd1765b7e                    
local               datamysql

您可以使用 docker inspect mycontainer 查看容器卷的确切位置

{
                "Type": "volume",
                "Name": "8d7a35354f666b2e8a26866a35bbae36bb9601701d4c6b505ab8ce6629f69415",
                "Source": "/media/USBdrive/docker/volumes/8d7a35354f666b2e8a26866a35bbae36bb9601701d4c6b505ab8ce6629f69415/_data",
                "Destination": "/var/lib/mysql",
                "Driver": "local",
                "Mode": "",
                "RW": true,
                "Propagation": ""
            },

删除未使用的卷可能很方便(特别是对于第三种情况)。

$ docker volume prune
WARNING! This will remove all volumes not used by at least one container.
Are you sure you want to continue? [y/N] y
Deleted Volumes:
4753ad0437d13e54c76d9c34a30a1843396a1866a0cf9237d500fdcca0d78c5f

Total reclaimed space: 205MB

因为您在 Dockerfile 中使用了 VOLUME 指令,所以您属于第三种情况。检查您的容器以查找文件,如果您希望重复 session 来持久保存数据,则从命令行指定卷。

关于postgresql - Docker volume 不持久化数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41777037/

相关文章:

更改函数以便它检索数组类型参数会导致错误 : FUNCTION DOES NOT EXIST

docker - 将环境变量传递给 dockerfile 构建

docker - 如何将Maven Jar文件添加到Docker容器

bash - Docker-Container内jHipster中的Maven包装器:FileNotFoundException

docker - Docker-与许多(数百)个环境组成

docker - 有没有办法创建 Docker 卷并用数据预填充它?

docker - Amazon ECR 的拉取访问被拒绝,存储库不存在或可能需要 'docker login'

postgresql - 如何更改通过 Google 控制台创建的数据库中公共(public)架构的所有者?

sql - SQL 查询内部如何执行 IN 表达式?

SQL 连接排序和限制结果