0

我用 dockerfile 做了一个简单的 hello world node.js 应用程序

FROM node:10
WORKDIR /usr/src/app
COPY package*.json ./

RUN npm install
COPY . .
EXPOSE 8080
CMD node index.js

如果我在我的 dockerfile 中将 CMD 更改为 RUN ,它仍然可以工作。在 dockerfile 中记录了使用 CMD,因为它将在容器运行时启动节点服务器。

我想知道如果我使用 RUN cmd 而不是 CMD 会发生什么。基本上,如果我制作一个本身处于运行状态的 docker 映像会发生什么。

4

3 回答 3

3

RUN将在构建过程中执行命令。在执行CMD容器时用作默认命令,而不是构建。如果您在指令中运行,您的构建将永远不会完成,并且您没有与他人共享的容器。node index.jsRUN

有关更多详细信息,请参阅 dockerfile 文档:RUNCMD.

该文档中的相关位:

RUN 指令将在当前图像之上的新层中执行任何命令并提交结果。生成的提交图像将用于 Dockerfile 中的下一步。

CMD 的主要目的是为执行容器提供默认值。


编辑:使用 OP 的index.jsonpackage.jsonDockerfile文件,docker 映像构建在使用时未完成,并且在使用时RUN node index.js完成(如预期)CMD node index.js

内容index.js

//Load express module with `require` directive
var express = require('express')
var app = express()

//Define request response in root URL (/)
app.get('/', function (req, res) {
  res.send('Hello World!')
})

//Launch listening server on port 8081
app.listen(8080, function () {
  console.log('app listening on port 8080!')
})

内容package.json

{
  "name": "dummy_nodejs_app",
  "version": "1.0.0",
  "description": "Node.js on Docker",
  "author": "Debojit",
  "main": "server.js",
  "scripts": {
    "start": "node server.js"
  },
  "dependencies": {
    "express": "^4.16.1"
  }
}

使用时Dockerfile如下:

FROM node:10
WORKDIR /usr/src/app
COPY package*.json ./

RUN npm install
COPY . .
EXPOSE 8080
RUN node index.js

然后构建挂起。这是输出:

jakub@dash:/tmp/test-node$ docker build -t test .
Sending build context to Docker daemon  4.096kB
Step 1/7 : FROM node:10
 ---> d5680e53a228
Step 2/7 : WORKDIR /usr/src/app
 ---> Using cache
 ---> a4b4547833e5
Step 3/7 : COPY package*.json ./
 ---> Using cache
 ---> 2b19cc3e48a3
Step 4/7 : RUN npm install
 ---> Using cache
 ---> fe1f1e72d17d
Step 5/7 : COPY . .
 ---> eb6fe0e3d1a7
Step 6/7 : EXPOSE 8080
 ---> Running in e573b923fcb2
Removing intermediate container e573b923fcb2
 ---> b3590153eed7
Step 7/7 : RUN node index.js
 ---> Running in 08b408e6e6f3
app listening on port 8080!

这无限期挂起。

使用 Dockerfile 时

FROM node:10
WORKDIR /usr/src/app
COPY package*.json ./

RUN npm install
COPY . .
EXPOSE 8080
CMD node index.js

构建输出是:

jakub@dash:/tmp/test-node$ docker build -t test .
Sending build context to Docker daemon  4.096kB
Step 1/7 : FROM node:10
 ---> d5680e53a228
Step 2/7 : WORKDIR /usr/src/app
 ---> Using cache
 ---> a4b4547833e5
Step 3/7 : COPY package*.json ./
 ---> Using cache
 ---> 2b19cc3e48a3
Step 4/7 : RUN npm install
 ---> Using cache
 ---> fe1f1e72d17d
Step 5/7 : COPY . .
 ---> Using cache
 ---> fc036f428e34
Step 6/7 : EXPOSE 8080
 ---> Using cache
 ---> d1ede7276d34
Step 7/7 : CMD node index.js
 ---> Using cache
 ---> cf051929395b
Successfully built cf051929395b
Successfully tagged test:latest
于 2019-12-12T15:30:27.073 回答
1

RUN步骤使用提供的命令执行临时容器,等待该命令退出,然后捕获对容器文件系统的更改作为生成图像的另一层。它不存储正在运行的进程、对环境变量的更改或对 shell 状态的任何更改,因为这些都没有写入文件系统。它也不会捕获对卷的更改,因为临时容器是使用映像中定义的卷启动的,并且对卷的更改不会应用于容器文件系统。这是一个构建时间步骤。

CMD步骤替换了当镜像作为容器运行时 docker 运行的现有默认命令。只要此命令正在运行,容器就会存在,并且您只能为该命令设置一个值。如果您CMD再次定义,则替换之前的值。如果您使用覆盖的命令启动容器,CMD则忽略图像值。

因此,您希望将在映像构建时修改文件系统的步骤RUNCMD分别在容器运行时执行的步骤分开。

于 2019-12-12T15:40:46.863 回答
0

首先,如果您在构建阶段使用RUN命令启动任何长时间运行的过程,您的构建过程将被卡住。

RUN命令在构建时执行,这是为构建时配置和安装包和工具而设计的,使用RUN命令准备 Docker 映像,例如安装 npm 模块和一些其他应用程序依赖项,这些依赖项将在进程启动时可用容器。

CMD启动容器时执行,它不会在构建时执行,应该CMD是一个长时间运行的过程来保留您的容器。

于 2019-12-12T15:36:47.857 回答