为什么 docker 创建空的 node_modules 以及如何避免它?

den*_*ski 8 node.js docker

有一些类似的问题,但他们没有回答为什么node_modulesdocker 在 repo 中创建空目录,即使 dockerfile 设置为在容器中保存 node_modules?

我很想知道为什么在主机上创建目录为空,假设纱线已经在容器内安装了包node_modules以及如何避免它。

## Dockerfile

FROM node:8.11.4-alpine
RUN apk update && apk add yarn
RUN yarn global add nodemon
WORKDIR /usr/app

COPY package.json yarn.lock /usr/app/
RUN yarn

EXPOSE 3000

Run Code Online (Sandbox Code Playgroud)
## docker-compose.yml

version: "3.2"

services:
  app:
    build:
      context: .
      dockerfile: Dockerfile
    command: nodemon index.js
    volumes:
      - .:/usr/app
      - /usr/app/node_modules
    ports:
      - "3000:3000"

Run Code Online (Sandbox Code Playgroud)

Mar*_*ark 19

你是对的,你的镜像构建过程是将节点包安装到node_modules镜像中的目录中。因此,在构建映像后,该映像包含node_modules并且您可以使用它来运行您的应用程序。

您可以node_modules在主机上看到,因为您在 Compose 文件中设置了卷。不过,它比您在其他答案中看到的要多。

发生的事情是您.:/usr/app在第一个卷定义中进行映射,这意味着您正在将主机上的当前目录映射到/usr/app容器中。

这将/usr/app使用主机上的当前目录覆盖图像中的目录。并且您的主机没有该node_modules目录(除非您也在主机上安装了 node_modules),因此您的容器将无法使用此映射,因为您已覆盖/usr/app并且覆盖中没有node_modules目录。Node 会抛出一个错误,说它找不到节点模块。

下一个volume mapping解决了这个问题,这其实是一个常见的Node开发设置。您创建一个卷/usr/app/node_modules。注意,这个卷没有:映射中没有的主机部分,这里只有一个目录。这意味着 Docker 将从/usr/app/node_modules映像挂载目录并将其添加到先前将主机目录映射到的映射中/usr/app

因此,在正在运行的容器中,node_modules由于双重映射,您将拥有来自主机当前目录的源代码以及来自底层映像的源代码。

作为副作用,您将node_modules在主机的当前目录中看到一个空目录。