我正在尝试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主机内,否则docker将创建一个目录。

例如:

/a/file/inside/host/hostFile.txt:/a/file/inside/container/containerFile.txt

hostFile.txt应该提前存在。 否则你会收到这个错误:containerFile.txt是一个目录

其他回答

TL;博士/注意: 如果您遇到在您试图挂载的文件的位置创建了一个目录,那么您可能无法提供有效的绝对路径。这是沉默和令人困惑的失败模式的常见错误。

文件卷在docker中是这样完成的(绝对路径示例(可以使用env变量),并且你需要提到文件名):

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

你还可以:

    volumes:
      - ${PWD}/upload:/var/www/html/upload
      - ${PWD}/upload/config.php:/var/www/html/config.php

如果你从/src/docker/myapp文件夹中触发docker-compose

在windows中, 如果你需要在docker-compose中使用a ${PWD} env变量。Yml,你可以在docker-compose的同一个目录下创建一个。env文件。Yml文件然后手动插入文件夹的位置。

CMD (pwd_var.bat):

echo PWD=%cd% >> .env

Powershell (pwd_var.ps1) :

$PSDefaultParameterValues['Out-File:Encoding'] = 'utf8'; echo "PWD=$(get-location).path" >> .env

docker-compose .env变量有更多好的特性: https://docs.docker.com/compose/reference/envvars/特别是COMPOSE_CONVERT_WINDOWS_PATHS env变量,它允许docker compose接受带有“\”的windows路径。

当您想要在windows上共享一个文件时,该文件必须在与容器共享之前存在。

TL&DR: Docker默认所有挂载点到目录。如果你想确保你的文件被挂载到docker实例中,这个文件必须在docker-compose启动和运行之前已经存在。

解释:

传统上,挂载卷的目的总是意味着将存储位置挂载到目录。然而,现在这种情况正在改变,现在可以共享文件了。从程序员的角度来看,很难知道“/tmp/something”是一个文件还是一个目录。因此,Docker假定所有挂载点首先是一个目录,除非文件已经存在。

在您的示例中,如果您希望挂载一个文件,请确保它已经存在于主机系统中。当Docker看到这个条目已经存在时,它会将这个文件挂载到Docker环境中。

如果您忘记在源目录中实际包含想要挂载的文件,似乎也会发生这种情况。我经历了惨痛的教训。没有诸如“文件不存在”之类的错误消息,而是在目标中创建一个目录。

我也曾有过类似的问题。我试图将我的配置文件导入到我的容器中,这样我就可以在每次需要时修复它,而无需重新构建映像。

我的意思是,我认为下面的命令会将$(pwd)/config.py从Docker主机映射到/root/app/config.py作为文件映射到容器中。

docker run -v $(pwd)/config.py:/root/app/config.py my_docker_image

但是,它总是创建一个名为config.py的目录,而不是一个文件。

在寻找线索的同时,我发现了原因(从这里)

如果使用-v或——volume来绑定挂载文件或目录 在Docker主机上还不存在,-v将为您创建端点。 它总是作为目录创建。

因此,它总是作为一个目录创建,因为我的docker主机没有$(pwd)/config.py。

即使我在docker主机中创建config.py。 $(pwd)/config.py只是覆盖了/root/app/config.py,而不是导出/root/app/config.py。