node.js - Node (Express & Mongo) 应用程序的生产与开发 Docker 设置

标签 node.js docker

我正在尝试将 Node 应用程序转换为使用 Docker,但遇到了一些我无法回答的问题。

但为了简单起见,我包含了一些非常基本的示例文件,以使问题保持​​在目标上。事实上,下面的示例只是链接到一个 Mongo 容器,但并没有在代码中使用它以使其更简单。

首先,需要什么 Dockerfiledocker-compose.yml 设置才能在本地 (OS X) 开发的 Node + Express + Mongo 应用程序上成功使用 Docker以及用于生产版本?

Dockerfile

FROM node:6.3.0

# Create new user to avoid using root - is this correct practise?
RUN useradd --user-group --create-home --shell /bin/false app

COPY package.json /home/app/code/
RUN chown -R app:app /home/app/*

USER app
WORKDIR /home/app/code
# Should this even be set here or use docker-compose instead?
# And should there be:
#  - docker-compose.yml setting it to production by default
#  - docker-compose.dev.yml setting it to production?
# Or reverse it? (docker-compose.prod.yml instead with default being development?)
# Commenting below out or it will always run as production
#ENV NODE_ENV production
RUN npm install

USER root
COPY . /home/app/code
# Running chown to ensure new 'app' user owns files
RUN chown -R app:app /home/app/*
USER app

EXPOSE 3000

# What CMD should be here to ensure development versus production is simple?
# Development - Restart server and refresh browser on file changes
# Production  - Ensure uptime. 
CMD ["npm", "start"]

docker-compose.yml

version: "2"
services:
  web:
    build: .
    # I would normally use a .env file but for this example will set explicitly
    # env_file: .env
    environment:
      - NODE_ENV=production
    volumes:
      - ./:/home/app/code
      - /home/app/code/node_modules
    ports:
      - "3000:3000"
    links:
      - mongo
  mongo:
    image: mongo
    ports:
      - "27017:27017"

docker-compose.dev.yml

version: "2"
services:
  web:
    # I would normally use a .env file but for this example will set explicitly
    # env_file: .env
    environment:
      - NODE_ENV=development

package.json

{
  "name": "docker-node-test",
  "version": "1.0.0",
  "description": "",
  "main": "app.js",
  "scripts": {
    "start": "nodemon app.js"
  },
  "dependencies": {
    "express": "^4.14.0",
    "mongoose": "^4.6.1",
    "nodemon": "^1.10.2"
  },
  "devDependencies": {
    "mocha": "^3.0.2"
  }
}

1。如何处理不同的 NODE_ENV(dev、production、staging)?

这是我的主要问题和难题。

在我使用的示例中,NODE_ENV 在 Dockerfile 中设置为 production,并且有两个 docker-compose 文件:

  • docker-compose.yml 将默认值包括 NODE_ENV 设置为 production
  • docker-compose.dev.yml 覆盖 NODE_ENV 并将其设置为 development

1.1. 是否建议切换该顺序并将开发设置作为默认设置,而不是使用 docker-compose.prod.yml 进行覆盖?

1.2.如何处理node_modules目录?

我真的不确定如何在本地开发需求和生产运行之间处理 node_modules 目录。 (也许我有一个根本的误解?)


编辑:

我添加了一个 .dockerignore 文件并将 node_modules 目录包含为一行。这样可以确保在复制过程中忽略 node_modules 目录等。

然后我编辑了 docker-compose.yml 以将 node_modules 包含为一个卷。

volumes:
  - ./:/home/app/code
  - /home/app/code/node_modules

为了完整起见,我还在问题开头将上述更改放入完整的 docker-compose.yml 中。

这甚至是一个解决方案吗?

执行上述操作可确保我的本地开发 npm install 包含开发依赖项。当运行 docker-compose up 时,它会在 Docker 容器中拉入仅用于生产的 Node 模块(因为默认的 docker-compose.yml 设置为 NODE_ENV=production)。

但是运行 docker-compose -f docker-compose.yml build 时,似乎没有考虑到 2 个 docker-compose 文件中设置的 NODE_ENV :/我希望它发送 NODE_ENV =生产,但所有 node_modules 都重新安装(包括开发依赖项)。

我们是否改为使用 2 个 Dockerfile? (用于生产的 Dockerfile;用于本地开发的 Dockerfile.dev)

(我觉得这是我在设置中缺少的基本逻辑/知识)


2。 Nodemon vs PM2

如何在本地开发机器上使用 nodemon 而在生产版本上使用 PM2?

3。您是否应该在 docker 容器中创建一个用户,然后将该用户设置为在 Dockerfile 中使用?

它默认使用 root 用户,但我没有看到很多文章谈论在容器中创建专用用户。我为安全所做的一切正确吗?我当然不会觉得在非 Docker 构建上以 root 身份运行应用程序。

感谢您的阅读。任何和所有的帮助表示赞赏:)

最佳答案

    1. 要么,没关系,我更喜欢有开发细节,然后用生产细节覆盖。

    2. 我没有将它们提交到我的仓库,然后我的 dockerfile 中有“npm install”。

  1. 您可以根据build设置在 dockerfile 中设置要构建的规则。

  2. 典型的做法是通过 root 构建一切,并通过 root 运行主程序。您可以设置其他用户,但在大多数情况下不需要这样做,因为 docker 容器的想法是将每个进程隔离在单独的 docker 容器中。

关于node.js - Node (Express & Mongo) 应用程序的生产与开发 Docker 设置,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39686643/

相关文章:

node.js - GraphQL 数据加载器与 Mongoose 填充

node.js - 登录成功后POST请求失败

node.js - "Failed to load resource: the server responded with a status of 404 (Not Found)"+ "Origin … is not allowed by Access-Control-Allow-Origin."

python - 为什么 conda install tk 在我的 docker 容器中不起作用,即使它说已安装?

docker - JBoss EAP 容器无法启动

javascript - 如何版本化和组织路由器 restify

node.js - Swagger:跳过未知类型 "dateTime"

mysql - docker-compose mysql 容器拒绝访问 wordpress 容器

docker - 如何在 dockerfile 中的 cd 命令之后执行 .sh 文件?

docker - Docker开机失败(vrc = VERR_NEM_VM_CREATE_FAILED,rc = E_FAIL(0X80004005)