3

我想构建一个将运行我的 node.js 应用程序的 Docker 映像。我正在使用dotenv npm 包来存储环境变量。.env位于项目的根目录中。当我运行npm run start_in_docker从主机启动应用程序()的 npm 脚本时,一切正常。但是,当我运行 Docker 容器时,出现以下错误:

internal/modules/cjs/loader.js:584
    throw err;
    ^

Error: Cannot find module 'dotenv'
    at Function.Module._resolveFilename (internal/modules/cjs/loader.js:582:15)
    at Function.Module._load (internal/modules/cjs/loader.js:508:25)
    at Module.require (internal/modules/cjs/loader.js:637:17)
    at require (internal/modules/cjs/helpers.js:22:18)
    at Object.<anonymous> (/usr/src/app/src/server/main.js:3:1)
    at Module._compile (internal/modules/cjs/loader.js:701:30)
    at Object.Module._extensions..js (internal/modules/cjs/loader.js:712:10)
    at Module.load (internal/modules/cjs/loader.js:600:32)
    at tryModuleLoad (internal/modules/cjs/loader.js:539:12)
    at Function.Module._load (internal/modules/cjs/loader.js:531:3) 

这是我的 Dockerfile:

FROM node:10-alpine

# Create app directory
WORKDIR /usr/src/app

# Install app dependencies
# A wildcard is used to ensure both package.json AND package-lock.json are copied
# where available (npm@5+)
COPY package*.json ./

RUN apk add --update python
RUN apk add --update vim
RUN apk add --update sqlite
RUN npm install --only=production

# Bundle app source
COPY . .

EXPOSE 3000

RUN npm install --unsafe-perm=true sqlite3 -g
RUN npm install pm2 -g
# run node-gyp to build sqlite3
RUN cd /usr/src/app/node_modules/sqlite3 && npm run install
WORKDIR /usr/src/app

这是我的 docker-compose.yml 文件,它运行上面的 Dockerfile:

version: '3'
services:
  appstore-bl-server:
    container_name: appstore-bl-server-production
    build:
      dockerfile: Dockerfile-prod
      context: .
    volumes:
      - ".:/usr/src/app"
    ports:
      - "3000:3000"
    networks:
      - appstore-net
    command: npm run start_in_docker

networks:
  appstore-net:
    driver: bridge

这是我的 package.json 脚本部分:

"scripts": {
    "start_in_docker": "NODE_ENV=development DEBUG=server:*,classes:*,middleware,utils node src/server/main.js",
}

这就是我dotenv在 main.js 文件中的要求:

require('dotenv').config();

我尝试指定绝对路径,('dotenv').config()但没有奏效。奇怪的是,如果我运行容器 shell ( docker run -it myimage /bin/ash) 然后运行npm run start_in_docker它也可以工作。dotenv当然在我的 package.json 中。

4

1 回答 1

2

我认为这是由于:

volumes:
  - ".:/usr/src/app"

在您的 docker-compose 文件中。这会将当前目录链接到 app 文件夹,这意味着包括所有内容都将node_modules被替换。

尝试从中删除卷docker-compose,看看它是否有效。

当您运行docker run -it myimage /bin/sh它时,它不会链接该卷,这意味着它将使用正确的node_modules,这就是它在使用时起作用的原因。

于 2019-05-11T15:19:29.357 回答