如何在Docker Compose中将主机目录挂载为卷


125

我有一个正在进行dockerizing的开发环境,并且我希望能够实时重新加载我的更改而不必重建docker映像。我使用docker compose是因为redis是我应用程序的依赖项之一,我喜欢能够链接redis容器

我在中定义了两个容器docker-compose.yml

node:
    build: ./node
    links:
        - redis
    ports:
        - "8080"
    env_file:
        - node-app.env

redis:
    image: redis
    ports:
        - "6379"

我已经到了我node应用程序的dockerfile中添加卷的地步,但是如何将主机的目录挂载到卷中,以便对代码的所有实时编辑都可以反映在容器中?

这是我当前的Dockerfile:

# Set the base image to Ubuntu
FROM    node:boron

# File Author / Maintainer
MAINTAINER Amin Shah Gilani <amin@gilani.me>

# Install nodemon
RUN npm install -g nodemon

# Add a /app volume
VOLUME ["/app"]

# TODO: link the current . to /app

# Define working directory
WORKDIR /app

# Run npm install
RUN npm install

# Expose port
EXPOSE  8080

# Run app using nodemon
CMD ["nodemon", "/app/app.js"]

我的项目如下所示:

/
- docker-compose.yml
- node-app.env
- node/
  - app.js
  - Dockerfile.js

Answers:


137

查看他们的文档

从外观上,您可以在docker-compose.yml上执行以下操作

volumes:
    - ./:/app

2
尝试了一下,但没有用:Error: Cannot find module '/data/app.js'
阿敏·沙·吉拉尼

2
等待..您的音量和WORKDIR指向/app/为什么您的CMD指向/data
jkris '16

1
那是一个错误!感谢您抓住这一点!已修复,现在我Error: Cannot find module 'express'要尝试一些操作。
阿敏·莎·吉拉尼

3
对于Windows容器,您将需要Windows样式路径,例如- .:c:/app(这使我绊倒了)
JohnnyFun

86

有几种选择

短语法

使用该host : guest格式,您可以执行以下任一操作:

volumes:
  # Just specify a path and let the Engine create a volume
  - /var/lib/mysql

  # Specify an absolute path mapping
  - /opt/data:/var/lib/mysql

  # Path on the host, relative to the Compose file
  - ./cache:/tmp/cache

  # User-relative path
  - ~/configs:/etc/configs/:ro

  # Named volume
  - datavolume:/var/lib/mysql

长语法

从docker-compose v3.2开始,您可以使用长语法,该语法允许配置可以以简短形式表示的其他字段,例如mount type(volume,bind或tmpfs)和read_only

version: "3.2"
services:
  web:
    image: nginx:alpine
    ports:
      - "80:80"
    volumes:
      - type: volume
        source: mydata
        target: /data
        volume:
          nocopy: true
      - type: bind
        source: ./static
        target: /opt/app/static

networks:
  webnet:

volumes:
  mydata:

查看https://docs.docker.com/compose/compose-file/#long-syntax-3了解更多信息。


21

如果要在Docker Compose YAML文件/disk1/prometheus-datavolumes部分中将特定的主机目录(在以下示例中)作为卷挂载,则可以按以下方式进行操作,例如:

version: '3'

services:
  prometheus:
    image: prom/prometheus
    volumes:
      - prometheus-data:/prometheus

volumes:
  prometheus-data:
    driver: local
    driver_opts:
      o: bind
      type: none
      device: /disk1/prometheus-data

顺便说一下,在普罗米修斯的Dockerfile中,您可能会发现 VOLUME指令,指令将其标记为从本地主机等保存外部安装的卷。(但是请注意:该指令不是必须的,但可以将卷安装到容器中。) :

Docker文件

...
VOLUME ["/prometheus"]
...

参考:


这工作了。谢谢。local驱动程序类型的文档在哪里?
梅维尔

@mmell该文档是第一个Refs链接(docs.docker.com/compose/compose-file/#driver),其内容为:指定应为此卷使用哪个卷驱动程序。默认为已配置为使用Docker引擎的任何驱动程序,大多数情况下为local
Yuci,

1
:@mmell,你可以找到这个问题的详细信息stackoverflow.com/questions/42195334/...
榆次

您怎么知道其中的设置driver_opts: o: bind type: none
mmell

1
@mmell基本上,这些选项与驱动程序有关,并且localLinux上的内置驱动程序接受类似于linux mount命令的选项:man7.org/linux/man-pages/man8/mount.8.html。你可以找到更多的讨论github.com/moby/moby/issues/19990#issuecomment-248955005,并stackoverflow.com/questions/35841241/...
榆次

9

有两件事:

我在中添加了音量docker-compose.yml

node:
    volumes:
        - ./node:/app

我将这些npm install && nodemon app.js片段移到了a中,CMD因为这RUN会将内容添加到Union File System中,并且我的卷不属于UFS。

# Set the base image to Ubuntu
FROM    node:boron

# File Author / Maintainer
MAINTAINER Amin Shah Gilani <amin@gilani.me>

# Install nodemon
RUN npm install -g nodemon

# Add a /app volume
VOLUME ["/app"]

# Define working directory
WORKDIR /app

# Expose port
EXPOSE  8080

# Run npm install
CMD npm install && nodemon app.js

2

我们必须先创建主机目录映射的自己的docker卷,然后在docker-compose.yml中将提及外部

1.创建名为共享的

docker volume create --driver local \
    --opt type=none \
    --opt device=/home/mukundhan/share \
    --opt o=bind share

2,在你的docker-compose中使用它

version: "3"

volumes:
  share:
    external: true

services:
  workstation:
    container_name: "workstation"
    image: "ubuntu"
    stdin_open: true
    tty: true
    volumes:
      - share:/share:consistent
      - ./source:/source:consistent
    working_dir: /source
    ipc: host
    privileged: true
    shm_size: '2gb'
  db:
    container_name: "db"
    image: "ubuntu"
    stdin_open: true
    tty: true
    volumes:
      - share:/share:consistent
    working_dir: /source
    ipc: host

这样,我们可以与在不同容器中运行的许多服务共享同一目录


为什么我需要ipc:host?
Vahid Noormofidi

仅在我们需要将网络绑定到主机本身时才需要。
Mukundhan
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.