我正在尝试dockerize一个PHP应用程序。在dockerfile中,我下载存档,提取它,等等。

一切都很好。但是,如果发布了一个新版本,并且我更新了dockerfile,我必须重新安装应用程序,因为config.php会被覆盖。

因此,我认为我可以将该文件作为卷挂载,就像对数据库所做的那样。

我尝试了两种方法,用音量和直接路径。

docker-compose:

version: '2'
services:
  app:
    build: src
    ports:
      - "8080:80"
    depends_on:
      - mysql
    volumes:
      -  app-conf:/var/www/html/upload
      -  app-conf:/var/www/html/config.php
    environment:
      DB_TYPE: mysql
      DB_MANAGER: MysqlManager

  mysql:
    image: mysql:5.6
    container_name: mysql
    volumes:
      - mysqldata:/var/lib/mysql
    ports:
      - 3306:3306
    environment:
      MYSQL_ROOT_PASSWORD:
      MYSQL_DATABASE:
      MYSQL_USER:
      MYSQL_PASSWORD:

volumes:
  mysqldata:
  app-conf:

这导致了错误:

我尝试用一个给定的路径,作为一个挂载的卷。

/src/docker/myapp/upload:/var/www/html/upload
/src/docker/myapp/upload:/var/www/html/config.php

然而,这两种方法都不起作用。对于挂载的卷,我看到创建了上传。

但它失败了:

/var/www/html/config.php\"造成\"不是目录\""

如果我用

/src/docker/myapp/upload/config.php:/var/www/html/config.php

Docker创建upload文件夹,然后创建config.php文件夹。不是文件。

或者是否有另一种方法来持久化配置?


当前回答

从docker-compose文件版本3.2开始,您可以指定类型为“bind”的卷挂载(而不是默认类型为“volume”),允许您将单个文件挂载到容器中。在docker-compose volume docs中搜索“bind mount”: https://docs.docker.com/compose/compose-file/#volumes

就我而言,我是在尝试发行一款单曲。”“Secrets”文件到我的应用程序中,其中只包含本地开发和测试的秘密。在生产环境中,我的应用程序从AWS中获取这些秘密。

如果我使用简写语法将这个文件挂载为卷:

volumes:
 - ./.secrets:/data/app/.secrets

Docker会创建一个“。容器内部的Secrets目录,而不是映射到容器外部的文件。然后,我的代码将引发一个错误,如“IsADirectoryError: [Errno 21] Is a directory: '.secrets'”。

我通过使用long-hand语法来解决这个问题,使用只读的“bind”卷挂载指定我的secrets文件:

volumes:
 - type: bind
   source: ./.secrets
   target: /data/app/.secrets
   read_only: true

现在Docker正确地将我的.secrets文件挂载到容器中,在容器中创建一个文件而不是一个目录。

其他回答

在合成中,我使用了一个相对路径,它工作:

version: "3.7"
services:
    svc1: 
      volumes:
        # Current dir is parent of src
        - "./src/file.conf:/path/in/container/file.conf 

使用docker run命令绑定挂载文件会产生:

docker: Error response from daemon: invalid mount config for type "bind": invalid mount path: 'path/file.conf' mount path must be absolute. 
See 'docker run --help'.

显然,这样做的唯一方法是指定一个绝对挂载路径,就像这样:

docker run -it --rm --mount type=bind,source="/path/to/file.conf",target=/file.conf alpine sh

对于Windows用户使用“%cd%”,对于Linux用户使用“$(pwd)”也是一种处理绝对路径的方法。

参见存储绑定挂载

对于Visual Studio Code用户,请确保您在命令提示符终端中运行%cd%命令,而不是PowerShell。

我在Windows上也遇到了同样的问题,Docker 18.66.1 -ce-win73(19507)。

通过Docker设置面板删除并重新添加共享驱动器,一切都恢复正常了。

使用mount(——mount)代替volume (-v)

更多信息:https://docs.docker.com/storage/bind-mounts/

例子:

确保docker主机上存在/tmp/a.txt

docker run -it --mount type=bind,source=/tmp/a.txt,target=/root/a.txt alpine sh

你也可以在docker-compose中使用相对路径。yml文件(在Windows主机,Linux容器上测试):

volumes:
    - ./test.conf:/fluentd/etc/test.conf

我的Windows 8.1也有同样的问题

结果证明,这是由于路径的大小写敏感性。 我调用docker-compose up从目录cd /c/users/alex/和容器内的一个文件被转换到目录。

但是当我执行cd /c/Users/alex/(不是大写的Users)并从那里调用docker-compose时,它工作了。

在我的系统中,Users dir和Alex dir都是大写的,尽管看起来只有Users dir重要。