查看原文
其他

Node.js 服务 Docker 容器化应用实践

五月君 Nodejs技术栈 2022-06-19

谁不会休息,谁就不会工作。—— 列宁

本篇不会讲解 Docker 命令的使用、安装等,因为在之前一篇文章 一文零基础教你学会 Docker 入门到实践 中也已经讲解的很详细了,不清楚的可以点击链接回头在重新看下,本篇重点是介绍 Node.js 项目如何进行 Docker 容器化及一些实践优化,还有一些常见的问题,当然如果还有其它使用上的问题也欢迎大家在评论区进行留言补充。

作者简介:五月君,Nodejs Developer,热爱技术、喜欢分享的 90 后青年,公众号「Nodejs技术栈」,Github 开源项目 https://www.nodejs.red

通过本篇文章能学到什么?

  • 学会如何用 Docker 容器化一个 Node.js 服务

  • 动态设置环境变量一份 Dockerfile 文件构建不同的版本

  • Node.js 私有 NPM 包在构建镜像时如何认证

  • Egg.js 框架 Docker 容器化应该注意的问题

  • Docker 镜像体积与构建时间的优化

Docker 化一个 Node.js 应用程序

在本篇开始我们先创建一个简单的 Node.js 应用,然后为这个应用创建一个 Docker 镜像,并构建和运行它

创建 Node.js 项目

首先我们需要创建一个 app.js 开启一个 HTTP 服务,后面会借助 Docker 来运行这个程序

  1. const http = require('http');

  2. const PORT = 30010;


  3. const server = http.createServer((req, res) => {

  4. res.end('Hello Docker');

  5. })


  6. server.listen(PORT, () => {

  7. console.log('Running on http://localhost:', PORT, 'NODE_ENV', process.env.NODE_ENV);

  8. });

然后我们创建一个 package.json 文件,这里是描述你的应用程序以及需要的依赖,写过 Node.js 的同学应该会很熟悉的,这里我在 scripts 里面增加了 npm run devnpm run pro 两个命令,因为我想在这里介绍如何在构建时传入参数来动态设置环境变量。

  1. {

  2. "name": "hello-docker",

  3. "version": "1.0.2",

  4. "description": "",

  5. "author": "May",

  6. "main": "app.js",

  7. "scripts": {

  8. "dev": "NODE_ENV=dev node app.js",

  9. "pro": "NODE_ENV=pro node app.js"

  10. }

  11. }

Dockerfile 文件

这是一个 Dockerfile 文件所包含的信息,这些命令在 Docker 入门与实践 中也有讲解过

  1. FROM node:10.0-alpine


  2. RUN apk --update add tzdata \

  3. && cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime \

  4. && echo "Asia/Shanghai" > /etc/timezone \

  5. && apk del tzdata


  6. RUN mkdir -p /usr/src/nodejs/


  7. WORKDIR /usr/src/nodejs/


  8. # add npm package

  9. COPY package.json /usr/src/nodejs/package.json

  10. RUN cd /usr/src/nodejs/

  11. RUN npm i


  12. # copy code

  13. COPY . /usr/src/nodejs/


  14. EXPOSE 30010


  15. CMD npm run dev

在 Dockerfile 的同级文件下创建一个 .dockerignore 文件,避免将你本地的调试文件、node_modules 等一些文件放入 Docker 容器中

  1. .git

  2. node_modules

  3. npm-debug.log

此时通过以下命令即可构建一个 Docker 镜像

  1. $ docker image build -t mayjun/hello-docker

再通过 docker run -d -p 30010:30010 mayjun/hello-docker 命令可运行一个 Docker 容器,但是有个疑问我是有生产和测试之分的,按照上面 CMD npm run dev 这样写死只能打包一种环境,当然你也可以在建一个文件来实现或者一些其它的方法。

动态设置环境变量

为了解决上面的疑问,我的想法是在镜像构建时传入参数来动态设置环境变量,对 Dockerfile 文件做下修改,看以下实现:

  1. EXPOSE 30010


  2. ARG node_env # 新增加

  3. ENV NODE_ENV=$node_env # 新增加

  4. CMD npm run ${NODE_ENV} # 修改

下面对上面的代码做个解释

  • 通过 ARG 指令定义了一个变量,用户可以在构建时通过使用 --build-arg = 标志的 docker build 命令将其传递给构建器 ARG node_env

  • 在 Dockerfile 中使用 ENV 引用这个变量 ENV NODE_ENV=$node_env

  • 这一步就是使用了 CMD npm run ${NODE_ENV}

剩下的就是在构建镜像时动态传入参数了

  1. $ docker image build --build-arg node_env=dev -t mayjun/hello-docker:1.0.2 . # 构建测试环境

  2. $ docker image build --build-arg node_env=pro -t mayjun/hello-docker:1.0.2 . # 构建生产环境

运行容器

  1. $ docker run -d -p 30010:30010 mayjun/hello-docker:1.0.2

  2. $ docker ps

  3. CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES

  4. 2bc6e62cd0e8 mayjun/hello-docker:1.0.2 "/bin/sh -c 'npm run…" 3 minutes ago Up 3 minutes 0.0.0.0:30010->30010/tcp elastic_bouman

查看容器日志

  1. docker logs -f 2bc6e62cd0e8


  2. > hello-docker@1.0.0 dev /usr/src/nodejs

  3. > NODE_ENV=dev node app.js


  4. Running on http://localhost: 30010 NODE_ENV dev

我将以上代码打包成了镜像 mayjun/hello-docker:1.0.2,可以拉取查看 docker pull mayjun/hello-docker:1.0.2

关于环境变量设置这只是一种方式,还可通过 docker-compose 来实现

Docker 与 Node.js 私有 NPM 包

如果你的项目中使用了私有 NPM 包,在 Dcoker 构建镜像过程中会出现 npm 私有包安装 404 的错误,如果是在容器外部我们可以 npm login 登陆拥有 NPM 私有包权限的账户,来解决这个问题,但是在 Docker 的时候是不能这样做的。

创建身份验证令牌

为了安装私有包我们需要 “创建身份验证令牌” 以便在持续集成环境、Docker 容器内部能访问我们的私有 NPM 包,如何创建可参考 https://docs.npmjs.com/creating-and-viewing-authentication-tokens

实现方法

我们在创建 Dockerfile 文件过程中就需要增加以下两条命令:

  1. # 528das62-e03e-4dc2-ba67-********** 这个 Token 就为你创建的身份验证令牌 token

  2. RUN echo "//registry.npmjs.org/:_authToken=528das62-e03e-4dc2-ba67-**********" > /root/.npmrc

  3. RUN cat /root/.npmrc

Egg 框架 Docker 容器化

在 Egg 里面,如果是 egg-scripts start--daemon去掉 --daemon 直接 egg-scripts start 即可,否则 Docker 容器会无法启动。

看以下代码示例,修改下 package.json 即可,Dockerfile 文件同上面第一个 Docker 化一个 Node.js 应用程序 是一样的

package.json

  1. {

  2. "scripts": {

  3. "start": "egg-scripts start" // 去掉 --daemon

  4. }

  5. }

也可参考 Egg Issues “docker容器不能run起来,请问有碰到的吗?” https://github.com/eggjs/egg/issues/1543

Docker 镜像体积与构建时间优化

如果一个镜像在不经过优化的情况下体积通常都是会很大的,以下也是在实践过程中做的几点优化。

RUN/COPY 分层

Dockerfile 中的每条指令都会创建一个镜像层,Dockerfile 指令或复制的项目文件在没有修改变动的情况下,每个镜像层是可以被复用和缓存的。

以下代码可在 mayjun/hello-docker:latest 镜像仓库找到,以下示例中,源码改变之后,不管 package.json 有没有改变的情况下都会重新安装 NPM 模块,这样显然是不好的,因此下面我们要改进

  1. # ...


  2. WORKDIR /usr/src/nodejs/hello-docker

  3. COPY . /usr/src/nodejs/hello-docker


  4. RUN npm install


  5. # ...

改进之后的代码如下所示,我们让 package.json 提前,在 package.json 没有修改的情况下是不会重新安装 NPM 包的,也会减少部署的时间。

  1. # ...


  2. WORKDIR /usr/src/nodejs/


  3. # add npm package

  4. COPY package.json /usr/src/app/package.json

  5. RUN cd /usr/src/app/

  6. RUN npm i


  7. # copy code

  8. COPY . /usr/src/app/


  9. # ...

Node.js Alpine 镜像优化

mayjun/hello-docker:1.0.0 这个镜像在 Docker 仓库也可搜索到,在未优化之前大约在 688MB

$ docker images REPOSITORY TAG IMAGE ID CREATED SIZE mayjun/hello-docker 1.0.0 7217fb3e9daa 5 seconds ago 688MB

使用 Alpine 优化

Alpine 是一个很小的 Linux 发行版,想要大幅度减小镜像体积选择 Node.js 的 Alpine 版本也是最简单的,另外 -alpine 的时区默认不是国内的,需要 Dockerfile 配置时区。

  1. FROM node:10.0-alpine


  2. RUN apk --update add tzdata \

  3. && cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime \

  4. && echo "Asia/Shanghai" > /etc/timezone \

  5. && apk del tzdata


  6. RUN echo "Asia/Shanghai" > /etc/timezone


  7. RUN mkdir -p /usr/src/nodejs/


  8. WORKDIR /usr/src/nodejs/


  9. # add npm package

  10. COPY package.json /usr/src/app/package.json

  11. RUN cd /usr/src/app/

  12. RUN npm i


  13. # copy code

  14. COPY . /usr/src/app/


  15. EXPOSE 30010

  16. CMD npm start

重新打包了一个版本 mayjun/hello-docker:1.1.0 再次查看下效果,可以看到镜像文件从 688MB 减少至 85.3MB,这个体积优化还是很大的

  1. $ docker images

  2. REPOSITORY TAG IMAGE ID CREATED SIZE

  3. mayjun/hello-docker 1.1.0 169e05b8197d 3 minutes ago 85.3MB

生产环境不要打包 devDependencies 包

有些测试环境用的包,在进行生产环境打镜像时不要包含进去,也就是 package.json 文件 devDependencies 对象,通过在 npm i 之后指定 --production 参数过滤

改进如下所示:

  1. FROM node:10.0-alpine


  2. # 省略 ...


  3. # add npm package

  4. COPY package.json /usr/src/app/package.json

  5. RUN cd /usr/src/app/

  6. RUN npm i --production # 改变在这了


  7. # 省略 ...

重新打包了一个版本 mayjun/hello-docker:1.2.0 再次查看下效果,可以看到镜像文件从 85.3MB 又减少至 72.3MB

  1. $ docker images

  2. REPOSITORY TAG IMAGE ID CREATED SIZE

  3. mayjun/hello-docker 1.2.0 f018aa578711 3 seconds ago 72.3MB

常见问题

Question1

以下命令在删除镜像的时候报如下错误:

  1. $ docker rmi 6b1c2775591e

  2. Error response from daemon: conflict: unable to delete 6b1c2775591e (must be forced) - image is referenced in multiple repositories

细心的你也许会发现镜像 ID 6b1c2775591e 同时指向了 hello-docker 和 mayjun/hello-docker 仓库,这也是造成删除失败的原因

  1. $ docker images

  2. REPOSITORY TAG IMAGE ID CREATED SIZE

  3. mysql 5.7 383867b75fd2 6 days ago 373MB

  4. hello-docker latest 6b1c2775591e 7 days ago 675MB

  5. mayjun/hello-docker latest 6b1c2775591e 7 days ago 675MB

指定 repository 和 tag 来删除,执行删除命令之后再次查看 mayjun/hello-docker 仓库就已经没有了

  1. $ docker rmi mayjun/hello-docker

  2. $ docker images

  3. REPOSITORY TAG IMAGE ID CREATED SIZE

  4. mysql 5.7 383867b75fd2 6 days ago 373MB

  5. hello-docker latest 6b1c2775591e 7 days ago 675MB

Question2

执行删除镜像命令报如下错误:

  1. $ docker rmi 9be467fd1285

  2. Error response from daemon: conflict: unable to delete 9be467fd1285 (cannot be forced) - image is being used by running container 1febfb05b850

根据提示是有正在运行的容器,需先停止容器、删除容器之后在删除镜像

  1. $ docker container kill 1febfb05b850 # 停止容器

  2. $ docker rm 1febfb05b850 # 删除容器

  3. $ docker rmi 9be467fd1285 # 删除镜像

Question3

设定的工作目录(WORKDIR)要与下面的要保持一致

  1. ...

  2. WORKDIR /usr/src/nodejs/


  3. # add npm package

  4. COPY package.json /usr/src/node/package.json # 目录不一致

  5. RUN cd /usr/src/node/ # 目录不一致

  6. RUN npm i

  7. ...

例如,如以上配置因为工作目录与实际 COPY 的目录不一致,会导致报以下错误:

再按照以下方式更改为一致即可

  1. ...

  2. WORKDIR /usr/src/nodejs/


  3. # add npm package

  4. COPY package.json /usr/src/nodejs/package.json # 更改为一致

  5. RUN cd /usr/src/nodejs/ # 更改为一致

  6. RUN npm i

  7. ...


往期精彩回顾
干货 | Node.js 在转转的微服务实践(一)
独家解读:淘宝使用 Node.js 的 TypeScript 多场景开发和实践
浅谈 Node.js 模块机制及常见面试问题解答
分享 10 道 Nodejs 进程相关面试题
Node.js 是什么?我为什么选择它?
数据结构知否知否系列之 — 栈篇
一文零基础教你学会 Docker 入门到实践
爱奇艺 PC Web Node.js 中间层实践
苏宁的Node.js实践:不低于Java的渲染性能、安全稳定迭代快
蚂蚁金服 Node.js 开荒史 - 摸爬滚打才不负功名尘土


好看你就点再看

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存